Files
allwpilib/src/main/native/cpp/UsbCameraImpl.h

184 lines
5.5 KiB
C
Raw Normal View History

2016-09-16 17:57:17 -07:00
/*----------------------------------------------------------------------------*/
2017-08-25 17:48:06 -07:00
/* Copyright (c) 2016-2017 FIRST. All Rights Reserved. */
2016-09-16 17:57:17 -07:00
/* Open Source Software - may be modified and shared by FRC teams. The code */
/* must be accompanied by the FIRST BSD license file in the root directory of */
/* the project. */
/*----------------------------------------------------------------------------*/
2017-08-25 17:48:06 -07:00
#ifndef CSCORE_USBCAMERAIMPL_H_
#define CSCORE_USBCAMERAIMPL_H_
2016-09-16 17:57:17 -07:00
#ifdef __linux__
#include <linux/videodev2.h>
#endif
2017-08-25 17:48:06 -07:00
#include <atomic>
#include <memory>
#include <string>
#include <thread>
#include <utility>
#include <vector>
#include <llvm/STLExtras.h>
#include <llvm/SmallVector.h>
#include <llvm/raw_ostream.h>
#include <support/condition_variable.h>
#include <support/mutex.h>
2017-08-25 17:48:06 -07:00
#include <support/raw_istream.h>
2016-09-16 17:57:17 -07:00
#include "SourceImpl.h"
#include "UsbCameraBuffer.h"
2016-12-22 22:08:08 -08:00
#include "UsbCameraProperty.h"
2016-09-16 17:57:17 -07:00
namespace cs {
class UsbCameraImpl : public SourceImpl {
2016-09-16 17:57:17 -07:00
public:
UsbCameraImpl(llvm::StringRef name, llvm::StringRef path);
~UsbCameraImpl() override;
2016-09-16 17:57:17 -07:00
void Start();
2016-09-16 17:57:17 -07:00
// Property functions
void SetProperty(int property, int value, CS_Status* status) override;
void SetStringProperty(int property, llvm::StringRef value,
CS_Status* status) override;
2016-09-16 17:57:17 -07:00
// Standard common camera properties
void SetBrightness(int brightness, CS_Status* status) override;
int GetBrightness(CS_Status* status) const override;
void SetWhiteBalanceAuto(CS_Status* status) override;
void SetWhiteBalanceHoldCurrent(CS_Status* status) override;
void SetWhiteBalanceManual(int value, CS_Status* status) override;
void SetExposureAuto(CS_Status* status) override;
void SetExposureHoldCurrent(CS_Status* status) override;
void SetExposureManual(int value, CS_Status* status) override;
bool SetVideoMode(const VideoMode& mode, CS_Status* status) override;
bool SetPixelFormat(VideoMode::PixelFormat pixelFormat,
CS_Status* status) override;
bool SetResolution(int width, int height, CS_Status* status) override;
bool SetFPS(int fps, CS_Status* status) override;
void NumSinksChanged() override;
void NumSinksEnabledChanged() override;
2016-09-16 17:57:17 -07:00
std::string GetPath() { return m_path; }
// Messages passed to/from camera thread
struct Message {
2016-11-15 23:54:50 -08:00
enum Kind {
kNone = 0,
kCmdSetMode,
kCmdSetPixelFormat,
kCmdSetResolution,
kCmdSetFPS,
kCmdSetProperty,
kCmdSetPropertyStr,
2017-08-25 17:48:06 -07:00
kNumSinksChanged, // no response
kNumSinksEnabledChanged, // no response
// Responses
kOk,
kError
};
2017-08-25 17:48:06 -07:00
explicit Message(Kind kind_)
: kind(kind_), from(std::this_thread::get_id()) {}
2016-11-15 23:54:50 -08:00
Kind kind;
int data[4];
std::string dataStr;
std::thread::id from;
};
2016-09-16 17:57:17 -07:00
protected:
std::unique_ptr<PropertyImpl> CreateEmptyProperty(
llvm::StringRef name) const override;
// Cache properties. Immediately successful if properties are already cached.
// If they are not, tries to connect to the camera to do so; returns false and
// sets status to CS_SOURCE_IS_DISCONNECTED if that too fails.
bool CacheProperties(CS_Status* status) const override;
private:
// Send a message to the camera thread and wait for a response (generic)
CS_StatusValue SendAndWait(Message&& msg) const;
// Send a message to the camera thread with no response
void Send(Message&& msg) const;
// The camera processing thread
2016-09-16 17:57:17 -07:00
void CameraThreadMain();
// Functions used by CameraThreadMain()
void DeviceDisconnect();
void DeviceConnect();
bool DeviceStreamOn();
bool DeviceStreamOff();
void DeviceProcessCommands();
void DeviceSetMode();
void DeviceSetFPS();
void DeviceCacheMode();
2016-12-22 22:08:08 -08:00
void DeviceCacheProperty(std::unique_ptr<UsbCameraProperty> rawProp);
void DeviceCacheProperties();
void DeviceCacheVideoModes();
// Command helper functions
CS_StatusValue DeviceProcessCommand(std::unique_lock<wpi::mutex>& lock,
const Message& msg);
CS_StatusValue DeviceCmdSetMode(std::unique_lock<wpi::mutex>& lock,
const Message& msg);
CS_StatusValue DeviceCmdSetProperty(std::unique_lock<wpi::mutex>& lock,
const Message& msg);
// Property helper functions
2016-12-22 22:08:08 -08:00
int RawToPercentage(const UsbCameraProperty& rawProp, int rawValue);
int PercentageToRaw(const UsbCameraProperty& rawProp, int percentValue);
void SetQuirks();
//
// Variables only used within camera thread
//
bool m_streaming;
bool m_modeSetPixelFormat{false};
bool m_modeSetResolution{false};
bool m_modeSetFPS{false};
#ifdef __linux__
unsigned m_capabilities = 0;
#endif
// Number of buffers to ask OS for
static constexpr int kNumBuffers = 4;
#ifdef __linux__
std::array<UsbCameraBuffer, kNumBuffers> m_buffers;
#endif
//
// Path never changes, so not protected by mutex.
//
2016-09-16 17:57:17 -07:00
std::string m_path;
#ifdef __linux__
2016-09-16 17:57:17 -07:00
std::atomic_int m_fd;
std::atomic_int m_command_fd; // for command eventfd
#endif
2016-09-16 17:57:17 -07:00
std::atomic_bool m_active; // set to false to terminate thread
2016-09-16 17:57:17 -07:00
std::thread m_cameraThread;
// Quirks
bool m_lifecam_exposure{false}; // Microsoft LifeCam exposure
//
// Variables protected by m_mutex
//
// Message queues
mutable std::vector<Message> m_commands;
mutable std::vector<std::pair<std::thread::id, CS_StatusValue>> m_responses;
mutable wpi::condition_variable m_responseCv;
2016-09-16 17:57:17 -07:00
};
} // namespace cs
2017-08-25 17:48:06 -07:00
#endif // CSCORE_USBCAMERAIMPL_H_