Commit 6c3c6032 authored by zork's avatar zork Committed by Commit bot

Move V4L2 code into a webcam helper class.

This will make it simpler to add new webcam interface types.

BUG=None

Review URL: https://codereview.chromium.org/1136883004

Cr-Commit-Position: refs/heads/master@{#329922}
parent d540b4c2
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "extensions/browser/api/webcam_private/v4l2_webcam.h"
#include <fcntl.h>
#include <linux/uvcvideo.h>
#include <linux/videodev2.h>
#include <stdio.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include "base/posix/eintr_wrapper.h"
#define V4L2_CID_PAN_SPEED (V4L2_CID_CAMERA_CLASS_BASE+32)
#define V4L2_CID_TILT_SPEED (V4L2_CID_CAMERA_CLASS_BASE+33)
#define V4L2_CID_PANTILT_CMD (V4L2_CID_CAMERA_CLASS_BASE+34)
// GUID of the Extension Unit for Logitech CC3300e motor control:
// {212de5ff-3080-2c4e-82d9-f587d00540bd}
#define UVC_GUID_LOGITECH_CC3000E_MOTORS \
{0x21, 0x2d, 0xe5, 0xff, 0x30, 0x80, 0x2c, 0x4e, \
0x82, 0xd9, 0xf5, 0x87, 0xd0, 0x05, 0x40, 0xbd}
#define LOGITECH_MOTORCONTROL_PANTILT_CMD 2
namespace {
const int kLogitechMenuIndexGoHome = 2;
const uvc_menu_info kLogitechCmdMenu[] = {
{1, "Set Preset"}, {2, "Get Preset"}, {3, "Go Home"}
};
const uvc_xu_control_mapping kLogitechCmdMapping = {
V4L2_CID_PANTILT_CMD,
"Pan/Tilt Go",
UVC_GUID_LOGITECH_CC3000E_MOTORS,
LOGITECH_MOTORCONTROL_PANTILT_CMD,
8,
0,
V4L2_CTRL_TYPE_MENU,
UVC_CTRL_DATA_TYPE_ENUM,
const_cast<uvc_menu_info*>(&kLogitechCmdMenu[0]),
arraysize(kLogitechCmdMenu),
};
} // namespace
namespace extensions {
V4L2Webcam::V4L2Webcam(const std::string& device_id) : device_id_(device_id) {
}
V4L2Webcam::~V4L2Webcam() {
}
bool V4L2Webcam::Open() {
fd_.reset(HANDLE_EINTR(open(device_id_.c_str(), 0)));
return fd_.is_valid();
}
bool V4L2Webcam::EnsureLogitechCommandsMapped() {
int res =
HANDLE_EINTR(ioctl(fd_.get(), UVCIOC_CTRL_MAP, &kLogitechCmdMapping));
// If mapping is successful or it's already mapped, this is a Logitech
// camera.
// NOTE: On success, occasionally EFAULT is returned. On a real error,
// ENOMEM, EPERM, EINVAL, or EOVERFLOW should be returned.
return res >= 0 || errno == EEXIST || errno == EFAULT;
}
bool V4L2Webcam::SetWebcamParameter(int fd, uint32_t control_id, int value) {
struct v4l2_control v4l2_ctrl = {control_id, value};
int res = HANDLE_EINTR(ioctl(fd, VIDIOC_S_CTRL, &v4l2_ctrl)) >= 0;
return res >= 0;
}
bool V4L2Webcam::GetWebcamParameter(int fd, uint32_t control_id, int* value) {
struct v4l2_control v4l2_ctrl = {control_id};
if (HANDLE_EINTR(ioctl(fd, VIDIOC_G_CTRL, &v4l2_ctrl)))
return false;
*value = v4l2_ctrl.value;
return true;
}
void V4L2Webcam::Reset(bool pan, bool tilt, bool zoom) {
if (pan || tilt) {
if (EnsureLogitechCommandsMapped()) {
SetWebcamParameter(fd_.get(), V4L2_CID_PANTILT_CMD,
kLogitechMenuIndexGoHome);
} else {
if (pan) {
struct v4l2_control v4l2_ctrl = {V4L2_CID_PAN_RESET};
HANDLE_EINTR(ioctl(fd_.get(), VIDIOC_S_CTRL, &v4l2_ctrl));
}
if (tilt) {
struct v4l2_control v4l2_ctrl = {V4L2_CID_TILT_RESET};
HANDLE_EINTR(ioctl(fd_.get(), VIDIOC_S_CTRL, &v4l2_ctrl));
}
}
}
if (zoom) {
const int kDefaultZoom = 100;
SetWebcamParameter(fd_.get(), V4L2_CID_ZOOM_ABSOLUTE, kDefaultZoom);
}
}
bool V4L2Webcam::GetPan(int* value) {
return GetWebcamParameter(fd_.get(), V4L2_CID_PAN_ABSOLUTE, value);
}
bool V4L2Webcam::GetTilt(int* value) {
return GetWebcamParameter(fd_.get(), V4L2_CID_TILT_ABSOLUTE, value);
}
bool V4L2Webcam::GetZoom(int* value) {
return GetWebcamParameter(fd_.get(), V4L2_CID_ZOOM_ABSOLUTE, value);
}
bool V4L2Webcam::SetPan(int value) {
return SetWebcamParameter(fd_.get(), V4L2_CID_PAN_ABSOLUTE, value);
}
bool V4L2Webcam::SetTilt(int value) {
return SetWebcamParameter(fd_.get(), V4L2_CID_TILT_ABSOLUTE, value);
}
bool V4L2Webcam::SetZoom(int value) {
return SetWebcamParameter(fd_.get(), V4L2_CID_ZOOM_ABSOLUTE, value);
}
bool V4L2Webcam::SetPanDirection(PanDirection direction) {
int direction_value = 0;
switch (direction) {
case PAN_STOP:
direction_value = 0;
break;
case PAN_RIGHT:
direction_value = 1;
break;
case PAN_LEFT:
direction_value = -1;
break;
}
return SetWebcamParameter(fd_.get(), V4L2_CID_PAN_SPEED, direction_value);
}
bool V4L2Webcam::SetTiltDirection(TiltDirection direction) {
int direction_value = 0;
switch (direction) {
case TILT_STOP:
direction_value = 0;
break;
case TILT_UP:
direction_value = 1;
break;
case TILT_DOWN:
direction_value = -1;
break;
}
return SetWebcamParameter(fd_.get(), V4L2_CID_TILT_SPEED, direction_value);
}
} // namespace extensions
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_V4L2_WEBCAM_H_
#define EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_V4L2_WEBCAM_H_
#include "extensions/browser/api/webcam_private/webcam.h"
#include "base/files/scoped_file.h"
namespace extensions {
class V4L2Webcam : public Webcam {
public:
V4L2Webcam(const std::string& device_id);
~V4L2Webcam() override;
bool Open();
private:
bool EnsureLogitechCommandsMapped();
bool SetWebcamParameter(int fd, uint32_t control_id, int value);
bool GetWebcamParameter(int fd, uint32_t control_id, int* value);
// Webcam:
void Reset(bool pan, bool tilt, bool zoom) override;
bool GetPan(int* value) override;
bool GetTilt(int* value) override;
bool GetZoom(int* value) override;
bool SetPan(int value) override;
bool SetTilt(int value) override;
bool SetZoom(int value) override;
bool SetPanDirection(PanDirection direction) override;
bool SetTiltDirection(TiltDirection direction) override;
const std::string device_id_;
base::ScopedFD fd_;
DISALLOW_COPY_AND_ASSIGN(V4L2Webcam);
};
} // namespace extensions
#endif // EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_V4L2_WEBCAM_H_
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "extensions/browser/api/webcam_private/webcam.h"
namespace extensions {
Webcam::Webcam() {}
Webcam::~Webcam() {}
} // namespace extensions
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_H_
#define EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_H_
#include <set>
#include <string>
#include "base/macros.h"
namespace extensions {
class Webcam {
public:
enum PanDirection {
PAN_LEFT,
PAN_RIGHT,
PAN_STOP,
};
enum TiltDirection {
TILT_UP,
TILT_DOWN,
TILT_STOP,
};
Webcam();
virtual ~Webcam();
virtual void Reset(bool pan, bool tilt, bool zoom) = 0;
virtual bool GetPan(int* value) = 0;
virtual bool GetTilt(int* value) = 0;
virtual bool GetZoom(int* value) = 0;
virtual bool SetPan(int value) = 0;
virtual bool SetTilt(int value) = 0;
virtual bool SetZoom(int value) = 0;
virtual bool SetPanDirection(PanDirection direction) = 0;
virtual bool SetTiltDirection(TiltDirection direction) = 0;
void AddExtensionRef(const std::string& extension_id) {
extension_refs_.insert(extension_id);
}
void RemoveExtensionRef(const std::string& extension_id) {
extension_refs_.erase(extension_id);
}
bool ShouldDelete() {
return extension_refs_.empty();
}
private:
std::set<std::string> extension_refs_;
DISALLOW_COPY_AND_ASSIGN(Webcam);
};
} // namespace extensions
#endif // EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_H_
...@@ -5,12 +5,65 @@ ...@@ -5,12 +5,65 @@
#ifndef EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_PRIVATE_API_H_ #ifndef EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_PRIVATE_API_H_
#define EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_PRIVATE_API_H_ #define EXTENSIONS_BROWSER_API_WEBCAM_PRIVATE_WEBCAM_PRIVATE_API_H_
#include <map>
#include "base/memory/scoped_ptr.h"
#include "base/scoped_observer.h"
#include "extensions/browser/browser_context_keyed_api_factory.h"
#include "extensions/browser/extension_function.h" #include "extensions/browser/extension_function.h"
#include "extensions/browser/process_manager_observer.h"
class Profile; class Profile;
namespace extensions { namespace extensions {
class ProcessManager;
class Webcam;
class WebcamPrivateAPI : public BrowserContextKeyedAPI,
public ProcessManagerObserver {
public:
static BrowserContextKeyedAPIFactory<WebcamPrivateAPI>* GetFactoryInstance();
// Convenience method to get the WebcamPrivateAPI for a BrowserContext.
static WebcamPrivateAPI* Get(content::BrowserContext* context);
explicit WebcamPrivateAPI(content::BrowserContext* context);
~WebcamPrivateAPI() override;
Webcam* GetWebcam(const std::string& extension_id,
const std::string& webcam_id);
private:
friend class BrowserContextKeyedAPIFactory<WebcamPrivateAPI>;
bool GetDeviceId(const std::string& extension_id,
const std::string& webcam_id,
std::string* device_id);
// ProcessManagerObserver:
void OnBackgroundHostClose(const std::string& extension_id) override;
// BrowserContextKeyedAPI:
static const char* service_name() {
return "WebcamPrivateAPI";
}
static const bool kServiceIsNULLWhileTesting = true;
static const bool kServiceRedirectedInIncognito = true;
content::BrowserContext* const browser_context_;
ScopedObserver<ProcessManager, ProcessManagerObserver>
process_manager_observer_;
std::map<std::string, linked_ptr<Webcam>> webcams_;
base::WeakPtrFactory<WebcamPrivateAPI> weak_ptr_factory_;
};
template <>
void BrowserContextKeyedAPIFactory<WebcamPrivateAPI>
::DeclareFactoryDependencies();
class WebcamPrivateSetFunction : public SyncExtensionFunction { class WebcamPrivateSetFunction : public SyncExtensionFunction {
public: public:
WebcamPrivateSetFunction(); WebcamPrivateSetFunction();
......
...@@ -4,33 +4,17 @@ ...@@ -4,33 +4,17 @@
#include "extensions/browser/api/webcam_private/webcam_private_api.h" #include "extensions/browser/api/webcam_private/webcam_private_api.h"
#include <fcntl.h> #include "base/lazy_instance.h"
#include <linux/uvcvideo.h>
#include <linux/videodev2.h>
#include <stdio.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include "base/files/scoped_file.h"
#include "base/posix/eintr_wrapper.h"
#include "content/public/browser/browser_context.h" #include "content/public/browser/browser_context.h"
#include "content/public/browser/media_device_id.h" #include "content/public/browser/media_device_id.h"
#include "content/public/browser/resource_context.h" #include "content/public/browser/resource_context.h"
#include "content/public/common/media_stream_request.h" #include "content/public/common/media_stream_request.h"
#include "extensions/browser/api/webcam_private/v4l2_webcam.h"
#include "extensions/browser/api/webcam_private/webcam.h"
#include "extensions/browser/process_manager.h"
#include "extensions/browser/process_manager_factory.h"
#include "extensions/common/api/webcam_private.h" #include "extensions/common/api/webcam_private.h"
#define V4L2_CID_PAN_SPEED (V4L2_CID_CAMERA_CLASS_BASE+32)
#define V4L2_CID_TILT_SPEED (V4L2_CID_CAMERA_CLASS_BASE+33)
#define V4L2_CID_PANTILT_CMD (V4L2_CID_CAMERA_CLASS_BASE+34)
// GUID of the Extension Unit for Logitech CC3300e motor control:
// {212de5ff-3080-2c4e-82d9-f587d00540bd}
#define UVC_GUID_LOGITECH_CC3000E_MOTORS \
{0x21, 0x2d, 0xe5, 0xff, 0x30, 0x80, 0x2c, 0x4e, \
0x82, 0xd9, 0xf5, 0x87, 0xd0, 0x05, 0x40, 0xbd}
#define LOGITECH_MOTORCONTROL_PANTILT_CMD 2
namespace webcam_private = extensions::core_api::webcam_private; namespace webcam_private = extensions::core_api::webcam_private;
namespace content { namespace content {
...@@ -38,70 +22,75 @@ class BrowserContext; ...@@ -38,70 +22,75 @@ class BrowserContext;
} // namespace content } // namespace content
namespace { namespace {
const int kLogitechMenuIndexGoHome = 2; const char kUnknownWebcam[] = "Unknown webcam id";
} // namespace
const uvc_menu_info kLogitechCmdMenu[] = {
{1, "Set Preset"}, {2, "Get Preset"}, {3, "Go Home"}
};
const uvc_xu_control_mapping kLogitechCmdMapping = {
V4L2_CID_PANTILT_CMD,
"Pan/Tilt Go",
UVC_GUID_LOGITECH_CC3000E_MOTORS,
LOGITECH_MOTORCONTROL_PANTILT_CMD,
8,
0,
V4L2_CTRL_TYPE_MENU,
UVC_CTRL_DATA_TYPE_ENUM,
const_cast<uvc_menu_info*>(&kLogitechCmdMenu[0]),
arraysize(kLogitechCmdMenu),
};
base::ScopedFD OpenWebcam(const std::string& extension_id,
content::BrowserContext* browser_context,
const std::string& webcam_id) {
GURL security_origin =
extensions::Extension::GetBaseURLFromExtensionId(extension_id);
std::string device_id;
bool success = content::GetMediaDeviceIDForHMAC(
content::MEDIA_DEVICE_VIDEO_CAPTURE,
browser_context->GetResourceContext()->GetMediaDeviceIDSalt(),
security_origin,
webcam_id,
&device_id);
if (!success) namespace extensions {
return base::ScopedFD();
return base::ScopedFD(HANDLE_EINTR(open(device_id.c_str(), 0))); // static
WebcamPrivateAPI* WebcamPrivateAPI::Get(content::BrowserContext* context) {
return GetFactoryInstance()->Get(context);
} }
void SetWebcamParameter(int fd, uint32_t control_id, int value) { WebcamPrivateAPI::WebcamPrivateAPI(content::BrowserContext* context)
struct v4l2_control v4l2_ctrl = {control_id, value}; : browser_context_(context),
HANDLE_EINTR(ioctl(fd, VIDIOC_S_CTRL, &v4l2_ctrl)); process_manager_observer_(this),
weak_ptr_factory_(this) {
process_manager_observer_.Add(ProcessManager::Get(browser_context_));
} }
bool GetWebcamParameter(int fd, uint32_t control_id, int* value) { WebcamPrivateAPI::~WebcamPrivateAPI() {}
struct v4l2_control v4l2_ctrl = {control_id};
if (HANDLE_EINTR(ioctl(fd, VIDIOC_G_CTRL, &v4l2_ctrl))) Webcam* WebcamPrivateAPI::GetWebcam(const std::string& extension_id,
return false; const std::string& webcam_id) {
std::string device_id;
if (!GetDeviceId(extension_id, webcam_id, &device_id)) {
return nullptr;
}
*value = v4l2_ctrl.value; auto ix = webcams_.find(device_id);
return true; if (ix != webcams_.end()) {
} ix->second->AddExtensionRef(extension_id);
return ix->second.get();
}
scoped_ptr<V4L2Webcam> v4l2_webcam(new V4L2Webcam(device_id));
if (!v4l2_webcam->Open()) {
return nullptr;
}
bool EnsureLogitechCommandsMapped(int fd) { linked_ptr<Webcam> webcam(v4l2_webcam.release());
int res = ioctl(fd, UVCIOC_CTRL_MAP, &kLogitechCmdMapping);
// If mapping is successful or it's already mapped, this is a Logitech camera. webcams_[device_id] = webcam;
return res >= 0 || errno == EEXIST; webcam->AddExtensionRef(extension_id);
return webcam.get();
} }
const char kUnknownWebcam[] = "Unknown webcam id"; bool WebcamPrivateAPI::GetDeviceId(const std::string& extension_id,
} // namespace const std::string& webcam_id,
std::string* device_id) {
GURL security_origin =
extensions::Extension::GetBaseURLFromExtensionId(extension_id);
namespace extensions { return content::GetMediaDeviceIDForHMAC(
content::MEDIA_DEVICE_VIDEO_CAPTURE,
browser_context_->GetResourceContext()->GetMediaDeviceIDSalt(),
security_origin,
webcam_id,
device_id);
}
void WebcamPrivateAPI::OnBackgroundHostClose(const std::string& extension_id) {
for (auto webcam = webcams_.begin();
webcam != webcams_.end(); /* No increment */ ) {
auto next = std::next(webcam);
webcam->second->RemoveExtensionRef(extension_id);
if (webcam->second->ShouldDelete())
webcams_.erase(webcam);
webcam = next;
}
}
WebcamPrivateSetFunction::WebcamPrivateSetFunction() { WebcamPrivateSetFunction::WebcamPrivateSetFunction() {
} }
...@@ -115,64 +104,61 @@ bool WebcamPrivateSetFunction::RunSync() { ...@@ -115,64 +104,61 @@ bool WebcamPrivateSetFunction::RunSync() {
webcam_private::Set::Params::Create(*args_)); webcam_private::Set::Params::Create(*args_));
EXTENSION_FUNCTION_VALIDATE(params.get()); EXTENSION_FUNCTION_VALIDATE(params.get());
base::ScopedFD fd = Webcam* webcam = WebcamPrivateAPI::Get(browser_context())->
OpenWebcam(extension_id(), browser_context(), params->webcam_id); GetWebcam(extension_id(), params->webcam_id);
if (!fd.is_valid()) { if (!webcam) {
SetError(kUnknownWebcam); SetError(kUnknownWebcam);
return false; return false;
} }
if (params->config.pan) { if (params->config.pan) {
SetWebcamParameter(fd.get(), V4L2_CID_PAN_ABSOLUTE, webcam->SetPan(*(params->config.pan));
*(params->config.pan));
} }
if (params->config.pan_direction) { if (params->config.pan_direction) {
int direction = 0; Webcam::PanDirection direction = Webcam::PAN_STOP;
switch (params->config.pan_direction) { switch (params->config.pan_direction) {
case webcam_private::PAN_DIRECTION_NONE: case webcam_private::PAN_DIRECTION_NONE:
case webcam_private::PAN_DIRECTION_STOP: case webcam_private::PAN_DIRECTION_STOP:
direction = 0; direction = Webcam::PAN_STOP;
break; break;
case webcam_private::PAN_DIRECTION_RIGHT: case webcam_private::PAN_DIRECTION_RIGHT:
direction = 1; direction = Webcam::PAN_RIGHT;
break; break;
case webcam_private::PAN_DIRECTION_LEFT: case webcam_private::PAN_DIRECTION_LEFT:
direction = -1; direction = Webcam::PAN_LEFT;
break; break;
} }
SetWebcamParameter(fd.get(), V4L2_CID_PAN_SPEED, direction); webcam->SetPanDirection(direction);
} }
if (params->config.tilt) { if (params->config.tilt) {
SetWebcamParameter(fd.get(), V4L2_CID_TILT_ABSOLUTE, webcam->SetTilt(*(params->config.tilt));
*(params->config.tilt));
} }
if (params->config.tilt_direction) { if (params->config.tilt_direction) {
int direction = 0; Webcam::TiltDirection direction = Webcam::TILT_STOP;
switch (params->config.tilt_direction) { switch (params->config.tilt_direction) {
case webcam_private::TILT_DIRECTION_NONE: case webcam_private::TILT_DIRECTION_NONE:
case webcam_private::TILT_DIRECTION_STOP: case webcam_private::TILT_DIRECTION_STOP:
direction = 0; direction = Webcam::TILT_STOP;
break; break;
case webcam_private::TILT_DIRECTION_UP: case webcam_private::TILT_DIRECTION_UP:
direction = 1; direction = Webcam::TILT_UP;
break; break;
case webcam_private::TILT_DIRECTION_DOWN: case webcam_private::TILT_DIRECTION_DOWN:
direction = -1; direction = Webcam::TILT_DOWN;
break; break;
} }
SetWebcamParameter(fd.get(), V4L2_CID_TILT_SPEED, direction); webcam->SetTiltDirection(direction);
} }
if (params->config.zoom) { if (params->config.zoom) {
SetWebcamParameter(fd.get(), V4L2_CID_ZOOM_ABSOLUTE, webcam->SetZoom(*(params->config.zoom));
*(params->config.zoom));
} }
...@@ -191,9 +177,9 @@ bool WebcamPrivateGetFunction::RunSync() { ...@@ -191,9 +177,9 @@ bool WebcamPrivateGetFunction::RunSync() {
webcam_private::Get::Params::Create(*args_)); webcam_private::Get::Params::Create(*args_));
EXTENSION_FUNCTION_VALIDATE(params.get()); EXTENSION_FUNCTION_VALIDATE(params.get());
base::ScopedFD fd = Webcam* webcam = WebcamPrivateAPI::Get(browser_context())->
OpenWebcam(extension_id(), browser_context(), params->webcam_id); GetWebcam(extension_id(), params->webcam_id);
if (!fd.is_valid()) { if (!webcam) {
SetError(kUnknownWebcam); SetError(kUnknownWebcam);
return false; return false;
} }
...@@ -201,15 +187,15 @@ bool WebcamPrivateGetFunction::RunSync() { ...@@ -201,15 +187,15 @@ bool WebcamPrivateGetFunction::RunSync() {
webcam_private::WebcamConfiguration result; webcam_private::WebcamConfiguration result;
int pan; int pan;
if (GetWebcamParameter(fd.get(), V4L2_CID_PAN_ABSOLUTE, &pan)) if (webcam->GetPan(&pan))
result.pan.reset(new double(pan)); result.pan.reset(new double(pan));
int tilt; int tilt;
if (GetWebcamParameter(fd.get(), V4L2_CID_TILT_ABSOLUTE, &tilt)) if (webcam->GetTilt(&tilt))
result.tilt.reset(new double(tilt)); result.tilt.reset(new double(tilt));
int zoom; int zoom;
if (GetWebcamParameter(fd.get(), V4L2_CID_ZOOM_ABSOLUTE, &zoom)) if (webcam->GetZoom(&zoom))
result.zoom.reset(new double(zoom)); result.zoom.reset(new double(zoom));
SetResult(result.ToValue().release()); SetResult(result.ToValue().release());
...@@ -229,36 +215,32 @@ bool WebcamPrivateResetFunction::RunSync() { ...@@ -229,36 +215,32 @@ bool WebcamPrivateResetFunction::RunSync() {
webcam_private::Reset::Params::Create(*args_)); webcam_private::Reset::Params::Create(*args_));
EXTENSION_FUNCTION_VALIDATE(params.get()); EXTENSION_FUNCTION_VALIDATE(params.get());
base::ScopedFD fd = Webcam* webcam = WebcamPrivateAPI::Get(browser_context())->
OpenWebcam(extension_id(), browser_context(), params->webcam_id); GetWebcam(extension_id(), params->webcam_id);
if (!fd.is_valid()) { if (!webcam) {
SetError(kUnknownWebcam); SetError(kUnknownWebcam);
return false; return false;
} }
if (params->config.pan || params->config.tilt) { webcam->Reset(params->config.pan, params->config.tilt, params->config.zoom);
if (EnsureLogitechCommandsMapped(fd.get())) {
SetWebcamParameter(fd.get(), V4L2_CID_PANTILT_CMD,
kLogitechMenuIndexGoHome);
}
}
if (params->config.pan) { return true;
struct v4l2_control v4l2_ctrl = {V4L2_CID_PAN_RESET}; }
HANDLE_EINTR(ioctl(fd.get(), VIDIOC_S_CTRL, &v4l2_ctrl));
}
if (params->config.tilt) { static base::LazyInstance<BrowserContextKeyedAPIFactory<WebcamPrivateAPI>>
struct v4l2_control v4l2_ctrl = {V4L2_CID_TILT_RESET}; g_factory = LAZY_INSTANCE_INITIALIZER;
HANDLE_EINTR(ioctl(fd.get(), VIDIOC_S_CTRL, &v4l2_ctrl));
}
if (params->config.zoom) { // static
const int kDefaultZoom = 100; BrowserContextKeyedAPIFactory<WebcamPrivateAPI>*
SetWebcamParameter(fd.get(), V4L2_CID_ZOOM_ABSOLUTE, kDefaultZoom); WebcamPrivateAPI::GetFactoryInstance() {
} return g_factory.Pointer();
}
return true; template <>
void BrowserContextKeyedAPIFactory<WebcamPrivateAPI>
::DeclareFactoryDependencies() {
DependsOn(ExtensionsBrowserClient::Get()->GetExtensionSystemFactory());
DependsOn(ProcessManagerFactory::GetInstance());
} }
} // namespace extensions } // namespace extensions
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
#include "extensions/browser/api/system_info/system_info_api.h" #include "extensions/browser/api/system_info/system_info_api.h"
#include "extensions/browser/api/usb/usb_event_router.h" #include "extensions/browser/api/usb/usb_event_router.h"
#include "extensions/browser/api/vpn_provider/vpn_service_factory.h" #include "extensions/browser/api/vpn_provider/vpn_service_factory.h"
#include "extensions/browser/api/webcam_private/webcam_private_api.h"
#include "extensions/browser/extension_message_filter.h" #include "extensions/browser/extension_message_filter.h"
#include "extensions/browser/extension_prefs_factory.h" #include "extensions/browser/extension_prefs_factory.h"
#include "extensions/browser/process_manager_factory.h" #include "extensions/browser/process_manager_factory.h"
...@@ -62,6 +63,9 @@ void EnsureBrowserContextKeyedServiceFactoriesBuilt() { ...@@ -62,6 +63,9 @@ void EnsureBrowserContextKeyedServiceFactoriesBuilt() {
StorageFrontend::GetFactoryInstance(); StorageFrontend::GetFactoryInstance();
SystemInfoAPI::GetFactoryInstance(); SystemInfoAPI::GetFactoryInstance();
UsbEventRouter::GetFactoryInstance(); UsbEventRouter::GetFactoryInstance();
#if defined(OS_CHROMEOS)
WebcamPrivateAPI::GetFactoryInstance();
#endif
} }
} // namespace extensions } // namespace extensions
...@@ -502,8 +502,6 @@ ...@@ -502,8 +502,6 @@
'browser/api/web_request/web_request_permissions.h', 'browser/api/web_request/web_request_permissions.h',
'browser/api/web_request/web_request_time_tracker.cc', 'browser/api/web_request/web_request_time_tracker.cc',
'browser/api/web_request/web_request_time_tracker.h', 'browser/api/web_request/web_request_time_tracker.h',
'browser/api/webcam_private/webcam_private_api.h',
'browser/api/webcam_private/webcam_private_api_chromeos.cc',
'browser/api_activity_monitor.h', 'browser/api_activity_monitor.h',
'browser/app_sorting.h', 'browser/app_sorting.h',
'browser/app_window/app_delegate.h', 'browser/app_window/app_delegate.h',
...@@ -799,6 +797,12 @@ ...@@ -799,6 +797,12 @@
'browser/api/vpn_provider/vpn_service.cc', 'browser/api/vpn_provider/vpn_service.cc',
'browser/api/vpn_provider/vpn_service.h', 'browser/api/vpn_provider/vpn_service.h',
'browser/api/vpn_provider/vpn_service_factory.h', 'browser/api/vpn_provider/vpn_service_factory.h',
'browser/api/webcam_private/v4l2_webcam.h',
'browser/api/webcam_private/v4l2_webcam.cc',
'browser/api/webcam_private/webcam.h',
'browser/api/webcam_private/webcam.cc',
'browser/api/webcam_private/webcam_private_api.h',
'browser/api/webcam_private/webcam_private_api_chromeos.cc',
], ],
'extensions_browser_sources_nonchromeos': [ 'extensions_browser_sources_nonchromeos': [
'browser/api/audio/audio_service.cc', 'browser/api/audio/audio_service.cc',
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment