cam: Make camera-related options sub-options of OptCamera

Use the new hierarchical options feature of the option parser to turn
camera-related option (--capture, --file, --stream, --strict-formats and
--metadata) into children of the --camera option. As an added bonus, we
don't need to check anymore if a camera has been specified when capture
is requested, as that's now enforced by the option parser.

This change prepares for support of multiple cameras.

Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Reviewed-by: Kieran Bingham <kieran.bingham@ideasonboard.com>
This commit is contained in:
Laurent Pinchart 2021-07-06 07:49:15 +03:00
parent 5082fe7b5b
commit 7dab1fa58f
3 changed files with 45 additions and 36 deletions

View file

@ -21,11 +21,11 @@
using namespace libcamera; using namespace libcamera;
CameraSession::CameraSession(CameraManager *cm, CameraSession::CameraSession(CameraManager *cm,
const std::string &cameraId,
const OptionsParser::Options &options) const OptionsParser::Options &options)
: last_(0), queueCount_(0), captureCount_(0), : options_(options), last_(0), queueCount_(0), captureCount_(0),
captureLimit_(0), printMetadata_(false) captureLimit_(0), printMetadata_(false)
{ {
const std::string &cameraId = options[OptCamera];
char *endptr; char *endptr;
unsigned long index = strtoul(cameraId.c_str(), &endptr, 10); unsigned long index = strtoul(cameraId.c_str(), &endptr, 10);
if (*endptr == '\0' && index > 0 && index <= cm->cameras().size()) if (*endptr == '\0' && index > 0 && index <= cm->cameras().size())
@ -44,7 +44,7 @@ CameraSession::CameraSession(CameraManager *cm,
return; return;
} }
StreamRoles roles = StreamKeyValueParser::roles(options[OptStream]); StreamRoles roles = StreamKeyValueParser::roles(options_[OptStream]);
std::unique_ptr<CameraConfiguration> config = std::unique_ptr<CameraConfiguration> config =
camera_->generateConfiguration(roles); camera_->generateConfiguration(roles);
@ -56,12 +56,12 @@ CameraSession::CameraSession(CameraManager *cm,
/* Apply configuration if explicitly requested. */ /* Apply configuration if explicitly requested. */
if (StreamKeyValueParser::updateConfiguration(config.get(), if (StreamKeyValueParser::updateConfiguration(config.get(),
options[OptStream])) { options_[OptStream])) {
std::cerr << "Failed to update configuration" << std::endl; std::cerr << "Failed to update configuration" << std::endl;
return; return;
} }
bool strictFormats = options.isSet(OptStrictFormats); bool strictFormats = options_.isSet(OptStrictFormats);
switch (config->validate()) { switch (config->validate()) {
case CameraConfiguration::Valid: case CameraConfiguration::Valid:
@ -134,14 +134,14 @@ void CameraSession::infoConfiguration() const
} }
} }
int CameraSession::start(const OptionsParser::Options &options) int CameraSession::start()
{ {
int ret; int ret;
queueCount_ = 0; queueCount_ = 0;
captureCount_ = 0; captureCount_ = 0;
captureLimit_ = options[OptCapture].toInteger(); captureLimit_ = options_[OptCapture].toInteger();
printMetadata_ = options.isSet(OptMetadata); printMetadata_ = options_.isSet(OptMetadata);
ret = camera_->configure(config_.get()); ret = camera_->configure(config_.get());
if (ret < 0) { if (ret < 0) {
@ -157,9 +157,9 @@ int CameraSession::start(const OptionsParser::Options &options)
camera_->requestCompleted.connect(this, &CameraSession::requestComplete); camera_->requestCompleted.connect(this, &CameraSession::requestComplete);
if (options.isSet(OptFile)) { if (options_.isSet(OptFile)) {
if (!options[OptFile].toString().empty()) if (!options_[OptFile].toString().empty())
writer_ = std::make_unique<BufferWriter>(options[OptFile]); writer_ = std::make_unique<BufferWriter>(options_[OptFile]);
else else
writer_ = std::make_unique<BufferWriter>(); writer_ = std::make_unique<BufferWriter>();
} }

View file

@ -9,6 +9,7 @@
#include <memory> #include <memory>
#include <stdint.h> #include <stdint.h>
#include <string>
#include <vector> #include <vector>
#include <libcamera/base/signal.h> #include <libcamera/base/signal.h>
@ -27,10 +28,12 @@ class CameraSession
{ {
public: public:
CameraSession(libcamera::CameraManager *cm, CameraSession(libcamera::CameraManager *cm,
const std::string &cameraId,
const OptionsParser::Options &options); const OptionsParser::Options &options);
~CameraSession(); ~CameraSession();
bool isValid() const { return config_ != nullptr; } bool isValid() const { return config_ != nullptr; }
const OptionsParser::Options &options() { return options_; }
libcamera::Camera *camera() { return camera_.get(); } libcamera::Camera *camera() { return camera_.get(); }
libcamera::CameraConfiguration *config() { return config_.get(); } libcamera::CameraConfiguration *config() { return config_.get(); }
@ -39,7 +42,7 @@ public:
void listProperties() const; void listProperties() const;
void infoConfiguration() const; void infoConfiguration() const;
int start(const OptionsParser::Options &options); int start();
void stop(); void stop();
libcamera::Signal<> captureDone; libcamera::Signal<> captureDone;
@ -51,6 +54,7 @@ private:
void requestComplete(libcamera::Request *request); void requestComplete(libcamera::Request *request);
void processRequest(libcamera::Request *request); void processRequest(libcamera::Request *request);
const OptionsParser::Options &options_;
std::shared_ptr<libcamera::Camera> camera_; std::shared_ptr<libcamera::Camera> camera_;
std::unique_ptr<libcamera::CameraConfiguration> config_; std::unique_ptr<libcamera::CameraConfiguration> config_;

View file

@ -113,19 +113,6 @@ int CamApp::parseOptions(int argc, char *argv[])
parser.addOption(OptCamera, OptionString, parser.addOption(OptCamera, OptionString,
"Specify which camera to operate on, by id or by index", "camera", "Specify which camera to operate on, by id or by index", "camera",
ArgumentRequired, "camera"); ArgumentRequired, "camera");
parser.addOption(OptCapture, OptionInteger,
"Capture until interrupted by user or until <count> frames captured",
"capture", ArgumentOptional, "count");
parser.addOption(OptFile, OptionString,
"Write captured frames to disk\n"
"If the file name ends with a '/', it sets the directory in which\n"
"to write files, using the default file name. Otherwise it sets the\n"
"full file path and name. The first '#' character in the file name\n"
"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", parser.addOption(OptHelp, OptionNone, "Display this help message",
"help"); "help");
parser.addOption(OptInfo, OptionNone, parser.addOption(OptInfo, OptionNone,
@ -138,12 +125,32 @@ int CamApp::parseOptions(int argc, char *argv[])
parser.addOption(OptMonitor, OptionNone, parser.addOption(OptMonitor, OptionNone,
"Monitor for hotplug and unplug camera events", "Monitor for hotplug and unplug camera events",
"monitor"); "monitor");
/* Sub-options of OptCamera: */
parser.addOption(OptCapture, OptionInteger,
"Capture until interrupted by user or until <count> frames captured",
"capture", ArgumentOptional, "count", false,
OptCamera);
parser.addOption(OptFile, OptionString,
"Write captured frames to disk\n"
"If the file name ends with a '/', it sets the directory in which\n"
"to write files, using the default file name. Otherwise it sets the\n"
"full file path and name. The first '#' character in the file name\n"
"is expanded to the stream name and frame sequence number.\n"
"The default file name is 'frame-#.bin'.",
"file", ArgumentOptional, "filename", false,
OptCamera);
parser.addOption(OptStream, &streamKeyValue,
"Set configuration of a camera stream", "stream", true,
OptCamera);
parser.addOption(OptStrictFormats, OptionNone, parser.addOption(OptStrictFormats, OptionNone,
"Do not allow requested stream format(s) to be adjusted", "Do not allow requested stream format(s) to be adjusted",
"strict-formats"); "strict-formats", ArgumentNone, nullptr, false,
OptCamera);
parser.addOption(OptMetadata, OptionNone, parser.addOption(OptMetadata, OptionNone,
"Print the metadata for completed requests", "Print the metadata for completed requests",
"metadata"); "metadata", ArgumentNone, nullptr, false,
OptCamera);
options_ = parser.parse(argc, argv); options_ = parser.parse(argc, argv);
if (!options_.valid()) if (!options_.valid())
@ -192,7 +199,10 @@ int CamApp::run()
std::unique_ptr<CameraSession> session; std::unique_ptr<CameraSession> session;
if (options_.isSet(OptCamera)) { if (options_.isSet(OptCamera)) {
session = std::make_unique<CameraSession>(cm_.get(), options_); const OptionValue &camera = options_[OptCamera];
session = std::make_unique<CameraSession>(cm_.get(),
camera.toString(),
camera.children());
if (!session->isValid()) { if (!session->isValid()) {
std::cout << "Failed to create camera session" << std::endl; std::cout << "Failed to create camera session" << std::endl;
return -EINVAL; return -EINVAL;
@ -223,13 +233,8 @@ int CamApp::run()
} }
/* 4. Start capture. */ /* 4. Start capture. */
if (options_.isSet(OptCapture)) { if (session && session->options().isSet(OptCapture)) {
if (!session) { ret = session->start();
std::cout << "Can't capture without a camera" << std::endl;
return -ENODEV;
}
ret = session->start(options_);
if (ret) { if (ret) {
std::cout << "Failed to start camera session" << std::endl; std::cout << "Failed to start camera session" << std::endl;
return ret; return ret;
@ -253,7 +258,7 @@ int CamApp::run()
loop_.exec(); loop_.exec();
/* 6. Stop capture. */ /* 6. Stop capture. */
if (options_.isSet(OptCapture)) if (session && session->options().isSet(OptCapture))
session->stop(); session->stop();
return 0; return 0;