libcamera/src/cam/main.cpp
Niklas Söderlund f28ca20960 libcamera: Use PixelFormat instead of unsigned int where appropriate
Use the PixelFormat instead of unsigned int where a pixel format is to
be used. PixelFormat is defined as an unsigned int but is about to be
turned into a class to add functionality.

There is no functional change in this patch.

Signed-off-by: Niklas Söderlund <niklas.soderlund@ragnatech.se>
Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
2020-03-18 18:02:40 +01:00

380 lines
8.2 KiB
C++

/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
* Copyright (C) 2019, Google Inc.
*
* main.cpp - cam - The libcamera swiss army knife
*/
#include <iomanip>
#include <iostream>
#include <signal.h>
#include <string.h>
#include <libcamera/libcamera.h>
#include <libcamera/property_ids.h>
#include "capture.h"
#include "event_loop.h"
#include "main.h"
#include "options.h"
using namespace libcamera;
class CamApp
{
public:
CamApp();
~CamApp();
static CamApp *instance();
int init(int argc, char **argv);
void cleanup();
int exec();
void quit();
private:
int parseOptions(int argc, char *argv[]);
int prepareConfig();
int listProperties();
int infoConfiguration();
int run();
static CamApp *app_;
OptionsParser::Options options_;
CameraManager *cm_;
std::shared_ptr<Camera> camera_;
std::unique_ptr<libcamera::CameraConfiguration> config_;
EventLoop *loop_;
};
CamApp *CamApp::app_ = nullptr;
CamApp::CamApp()
: cm_(nullptr), camera_(nullptr), config_(nullptr), loop_(nullptr)
{
CamApp::app_ = this;
}
CamApp::~CamApp()
{
delete cm_;
}
CamApp *CamApp::instance()
{
return CamApp::app_;
}
int CamApp::init(int argc, char **argv)
{
int ret;
ret = parseOptions(argc, argv);
if (ret < 0)
return ret;
cm_ = new CameraManager();
ret = cm_->start();
if (ret) {
std::cout << "Failed to start camera manager: "
<< strerror(-ret) << std::endl;
return ret;
}
if (options_.isSet(OptCamera)) {
const std::string &cameraName = options_[OptCamera];
char *endptr;
unsigned long index = strtoul(cameraName.c_str(), &endptr, 10);
if (*endptr == '\0' && index > 0 && index <= cm_->cameras().size())
camera_ = cm_->cameras()[index - 1];
else
camera_ = cm_->get(cameraName);
if (!camera_) {
std::cout << "Camera "
<< std::string(options_[OptCamera])
<< " not found" << std::endl;
cm_->stop();
return -ENODEV;
}
if (camera_->acquire()) {
std::cout << "Failed to acquire camera" << std::endl;
camera_.reset();
cm_->stop();
return -EINVAL;
}
std::cout << "Using camera " << camera_->name() << std::endl;
ret = prepareConfig();
if (ret)
return ret;
}
loop_ = new EventLoop(cm_->eventDispatcher());
return 0;
}
void CamApp::cleanup()
{
delete loop_;
loop_ = nullptr;
if (camera_) {
camera_->release();
camera_.reset();
}
config_.reset();
cm_->stop();
}
int CamApp::exec()
{
int ret;
ret = run();
cleanup();
return ret;
}
void CamApp::quit()
{
if (loop_)
loop_->exit();
}
int CamApp::parseOptions(int argc, char *argv[])
{
KeyValueParser streamKeyValue;
streamKeyValue.addOption("role", OptionString,
"Role for the stream (viewfinder, video, still)",
ArgumentRequired);
streamKeyValue.addOption("width", OptionInteger, "Width in pixels",
ArgumentRequired);
streamKeyValue.addOption("height", OptionInteger, "Height in pixels",
ArgumentRequired);
streamKeyValue.addOption("pixelformat", OptionInteger, "Pixel format",
ArgumentRequired);
OptionsParser parser;
parser.addOption(OptCamera, OptionString,
"Specify which camera to operate on, by name or by index", "camera",
ArgumentRequired, "camera");
parser.addOption(OptCapture, OptionNone,
"Capture until interrupted by user", "capture");
parser.addOption(OptFile, OptionString,
"Write captured frames to disk\n"
"The first '#' character in the file name is expanded to the stream name and frame sequence number.\n"
"The default file name is 'frame-#.bin'.",
"file", ArgumentOptional, "filename");
parser.addOption(OptStream, &streamKeyValue,
"Set configuration of a camera stream", "stream", true);
parser.addOption(OptHelp, OptionNone, "Display this help message",
"help");
parser.addOption(OptInfo, OptionNone,
"Display information about stream(s)", "info");
parser.addOption(OptList, OptionNone, "List all cameras", "list");
parser.addOption(OptProps, OptionNone, "List cameras properties",
"list-properties");
options_ = parser.parse(argc, argv);
if (!options_.valid())
return -EINVAL;
if (options_.empty() || options_.isSet(OptHelp)) {
parser.usage();
return options_.empty() ? -EINVAL : -EINTR;
}
return 0;
}
int CamApp::prepareConfig()
{
StreamRoles roles;
if (options_.isSet(OptStream)) {
const std::vector<OptionValue> &streamOptions =
options_[OptStream].toArray();
/* Use roles and get a default configuration. */
for (auto const &value : streamOptions) {
KeyValueParser::Options opt = value.toKeyValues();
if (!opt.isSet("role")) {
roles.push_back(StreamRole::VideoRecording);
} else if (opt["role"].toString() == "viewfinder") {
roles.push_back(StreamRole::Viewfinder);
} else if (opt["role"].toString() == "video") {
roles.push_back(StreamRole::VideoRecording);
} else if (opt["role"].toString() == "still") {
roles.push_back(StreamRole::StillCapture);
} else {
std::cerr << "Unknown stream role "
<< opt["role"].toString() << std::endl;
return -EINVAL;
}
}
} else {
/* If no configuration is provided assume a single video stream. */
roles.push_back(StreamRole::VideoRecording);
}
config_ = camera_->generateConfiguration(roles);
if (!config_ || config_->size() != roles.size()) {
std::cerr << "Failed to get default stream configuration"
<< std::endl;
return -EINVAL;
}
/* Apply configuration if explicitly requested. */
if (options_.isSet(OptStream)) {
const std::vector<OptionValue> &streamOptions =
options_[OptStream].toArray();
unsigned int i = 0;
for (auto const &value : streamOptions) {
KeyValueParser::Options opt = value.toKeyValues();
StreamConfiguration &cfg = config_->at(i++);
if (opt.isSet("width"))
cfg.size.width = opt["width"];
if (opt.isSet("height"))
cfg.size.height = opt["height"];
/* TODO: Translate 4CC string to ID. */
if (opt.isSet("pixelformat"))
cfg.pixelFormat = opt["pixelformat"];
}
}
switch (config_->validate()) {
case CameraConfiguration::Valid:
break;
case CameraConfiguration::Adjusted:
std::cout << "Camera configuration adjusted" << std::endl;
break;
case CameraConfiguration::Invalid:
std::cout << "Camera configuration invalid" << std::endl;
config_.reset();
return -EINVAL;
}
return 0;
}
int CamApp::listProperties()
{
if (!camera_) {
std::cout << "Cannot list properties without a camera"
<< std::endl;
return -EINVAL;
}
for (const auto &prop : camera_->properties()) {
const ControlId *id = properties::properties.at(prop.first);
const ControlValue &value = prop.second;
std::cout << "Property: " << id->name() << " = " << value.toString();
}
return 0;
}
int CamApp::infoConfiguration()
{
if (!config_) {
std::cout << "Cannot print stream information without a camera"
<< std::endl;
return -EINVAL;
}
unsigned int index = 0;
for (const StreamConfiguration &cfg : *config_) {
std::cout << index << ": " << cfg.toString() << std::endl;
const StreamFormats &formats = cfg.formats();
for (PixelFormat pixelformat : formats.pixelformats()) {
std::cout << " * Pixelformat: 0x" << std::hex
<< std::setw(8) << pixelformat << " "
<< formats.range(pixelformat).toString()
<< std::endl;
for (const Size &size : formats.sizes(pixelformat))
std::cout << " - " << size.toString()
<< std::endl;
}
index++;
}
return 0;
}
int CamApp::run()
{
int ret;
if (options_.isSet(OptList)) {
std::cout << "Available cameras:" << std::endl;
unsigned int index = 1;
for (const std::shared_ptr<Camera> &cam : cm_->cameras()) {
std::cout << index << ": " << cam->name() << std::endl;
index++;
}
}
if (options_.isSet(OptProps)) {
ret = listProperties();
if (ret)
return ret;
}
if (options_.isSet(OptInfo)) {
ret = infoConfiguration();
if (ret)
return ret;
}
if (options_.isSet(OptCapture)) {
Capture capture(camera_, config_.get());
return capture.run(loop_, options_);
}
return 0;
}
void signalHandler(int signal)
{
std::cout << "Exiting" << std::endl;
CamApp::instance()->quit();
}
int main(int argc, char **argv)
{
CamApp app;
int ret;
ret = app.init(argc, argv);
if (ret)
return ret == -EINTR ? 0 : EXIT_FAILURE;
struct sigaction sa = {};
sa.sa_handler = &signalHandler;
sigaction(SIGINT, &sa, nullptr);
if (app.exec())
return EXIT_FAILURE;
return 0;
}