This source file includes following definitions.
- NumberOfFakeDevices
- GetDeviceNames
- GetDeviceSupportedFormats
- Create
- SetFailNextCreate
- SetNumberOfFakeDevices
- format_roster_index_
- AllocateAndStart
- StopAndDeAllocate
- OnAllocateAndStart
- OnStopAndDeAllocate
- OnCaptureTask
- Reallocate
- PopulateFormatRoster
#include "media/video/capture/fake_video_capture_device.h"
#include <string>
#include "base/bind.h"
#include "base/memory/scoped_ptr.h"
#include "base/strings/stringprintf.h"
#include "media/audio/fake_audio_input_stream.h"
#include "media/base/video_frame.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/skia/include/core/SkCanvas.h"
#include "third_party/skia/include/core/SkPaint.h"
namespace media {
static const int kFakeCaptureTimeoutMs = 50;
static const int kFakeCaptureBeepCycle = 10;
static const int kFakeCaptureCapabilityChangePeriod = 30;
enum { kNumberOfFakeDevices = 2 };
bool FakeVideoCaptureDevice::fail_next_create_ = false;
base::subtle::Atomic32 FakeVideoCaptureDevice::number_of_devices_ =
kNumberOfFakeDevices;
size_t FakeVideoCaptureDevice::NumberOfFakeDevices(void) {
return number_of_devices_;
}
void FakeVideoCaptureDevice::GetDeviceNames(Names* const device_names) {
device_names->erase(device_names->begin(), device_names->end());
int number_of_devices = base::subtle::NoBarrier_Load(&number_of_devices_);
for (int32 n = 0; n < number_of_devices; n++) {
Name name(base::StringPrintf("fake_device_%d", n),
base::StringPrintf("/dev/video%d", n));
device_names->push_back(name);
}
}
void FakeVideoCaptureDevice::GetDeviceSupportedFormats(
const Name& device,
VideoCaptureFormats* supported_formats) {
const size_t supported_sizes_length = 3;
const gfx::Size supported_sizes[] = {gfx::Size(320, 240),
gfx::Size(640, 480),
gfx::Size(1280, 720)};
int frame_rate = 1000 / kFakeCaptureTimeoutMs;
supported_formats->clear();
for (size_t i=0; i < supported_sizes_length; ++i) {
supported_formats->push_back(VideoCaptureFormat(supported_sizes[i],
frame_rate,
media::PIXEL_FORMAT_I420));
}
}
VideoCaptureDevice* FakeVideoCaptureDevice::Create(const Name& device_name) {
if (fail_next_create_) {
fail_next_create_ = false;
return NULL;
}
int number_of_devices = base::subtle::NoBarrier_Load(&number_of_devices_);
for (int32 n = 0; n < number_of_devices; ++n) {
std::string possible_id = base::StringPrintf("/dev/video%d", n);
if (device_name.id().compare(possible_id) == 0) {
return new FakeVideoCaptureDevice();
}
}
return NULL;
}
void FakeVideoCaptureDevice::SetFailNextCreate() {
fail_next_create_ = true;
}
void FakeVideoCaptureDevice::SetNumberOfFakeDevices(size_t number_of_devices) {
base::subtle::NoBarrier_AtomicExchange(&number_of_devices_,
number_of_devices);
}
FakeVideoCaptureDevice::FakeVideoCaptureDevice()
: capture_thread_("CaptureThread"),
frame_count_(0),
format_roster_index_(0) {}
FakeVideoCaptureDevice::~FakeVideoCaptureDevice() {
DCHECK(thread_checker_.CalledOnValidThread());
DCHECK(!capture_thread_.IsRunning());
}
void FakeVideoCaptureDevice::AllocateAndStart(
const VideoCaptureParams& params,
scoped_ptr<VideoCaptureDevice::Client> client) {
DCHECK(thread_checker_.CalledOnValidThread());
DCHECK(!capture_thread_.IsRunning());
capture_thread_.Start();
capture_thread_.message_loop()->PostTask(
FROM_HERE,
base::Bind(&FakeVideoCaptureDevice::OnAllocateAndStart,
base::Unretained(this),
params,
base::Passed(&client)));
}
void FakeVideoCaptureDevice::StopAndDeAllocate() {
DCHECK(thread_checker_.CalledOnValidThread());
DCHECK(capture_thread_.IsRunning());
capture_thread_.message_loop()->PostTask(
FROM_HERE,
base::Bind(&FakeVideoCaptureDevice::OnStopAndDeAllocate,
base::Unretained(this)));
capture_thread_.Stop();
}
void FakeVideoCaptureDevice::OnAllocateAndStart(
const VideoCaptureParams& params,
scoped_ptr<VideoCaptureDevice::Client> client) {
DCHECK_EQ(capture_thread_.message_loop(), base::MessageLoop::current());
client_ = client.Pass();
DCHECK_EQ(params.requested_format.pixel_format, PIXEL_FORMAT_I420);
capture_format_.pixel_format = params.requested_format.pixel_format;
capture_format_.frame_rate = 30;
if (params.requested_format.frame_size.width() > 640)
capture_format_.frame_size.SetSize(1280, 720);
else if (params.requested_format.frame_size.width() > 320)
capture_format_.frame_size.SetSize(640, 480);
else
capture_format_.frame_size.SetSize(320, 240);
if (params.allow_resolution_change)
PopulateFormatRoster();
const size_t fake_frame_size =
VideoFrame::AllocationSize(VideoFrame::I420, capture_format_.frame_size);
fake_frame_.reset(new uint8[fake_frame_size]);
capture_thread_.message_loop()->PostTask(
FROM_HERE,
base::Bind(&FakeVideoCaptureDevice::OnCaptureTask,
base::Unretained(this)));
}
void FakeVideoCaptureDevice::OnStopAndDeAllocate() {
DCHECK_EQ(capture_thread_.message_loop(), base::MessageLoop::current());
client_.reset();
}
void FakeVideoCaptureDevice::OnCaptureTask() {
if (!client_)
return;
const size_t frame_size =
VideoFrame::AllocationSize(VideoFrame::I420, capture_format_.frame_size);
memset(fake_frame_.get(), 0, frame_size);
SkBitmap bitmap;
bitmap.setConfig(SkBitmap::kA8_Config,
capture_format_.frame_size.width(),
capture_format_.frame_size.height(),
capture_format_.frame_size.width()),
bitmap.setPixels(fake_frame_.get());
SkCanvas canvas(bitmap);
int radius = std::min(capture_format_.frame_size.width(),
capture_format_.frame_size.height()) / 4;
SkRect rect =
SkRect::MakeXYWH(capture_format_.frame_size.width() / 2 - radius,
capture_format_.frame_size.height() / 2 - radius,
2 * radius,
2 * radius);
SkPaint paint;
paint.setStyle(SkPaint::kFill_Style);
paint.setAlpha(128);
int end_angle = (frame_count_ % kFakeCaptureBeepCycle * 360) /
kFakeCaptureBeepCycle;
if (!end_angle)
end_angle = 360;
canvas.drawArc(rect, 0, end_angle, true, paint);
int elapsed_ms = kFakeCaptureTimeoutMs * frame_count_;
int milliseconds = elapsed_ms % 1000;
int seconds = (elapsed_ms / 1000) % 60;
int minutes = (elapsed_ms / 1000 / 60) % 60;
int hours = (elapsed_ms / 1000 / 60 / 60) % 60;
std::string time_string =
base::StringPrintf("%d:%02d:%02d:%03d %d", hours, minutes,
seconds, milliseconds, frame_count_);
canvas.scale(3, 3);
canvas.drawText(time_string.data(), time_string.length(), 30, 20,
paint);
if (frame_count_ % kFakeCaptureBeepCycle == 0) {
FakeAudioInputStream::BeepOnce();
}
frame_count_++;
client_->OnIncomingCapturedData(fake_frame_.get(),
frame_size,
capture_format_,
0,
base::TimeTicks::Now());
if (!(frame_count_ % kFakeCaptureCapabilityChangePeriod) &&
format_roster_.size() > 0U) {
Reallocate();
}
capture_thread_.message_loop()->PostDelayedTask(
FROM_HERE,
base::Bind(&FakeVideoCaptureDevice::OnCaptureTask,
base::Unretained(this)),
base::TimeDelta::FromMilliseconds(kFakeCaptureTimeoutMs));
}
void FakeVideoCaptureDevice::Reallocate() {
DCHECK_EQ(capture_thread_.message_loop(), base::MessageLoop::current());
capture_format_ =
format_roster_.at(++format_roster_index_ % format_roster_.size());
DCHECK_EQ(capture_format_.pixel_format, PIXEL_FORMAT_I420);
DVLOG(3) << "Reallocating FakeVideoCaptureDevice, new capture resolution "
<< capture_format_.frame_size.ToString();
const size_t fake_frame_size =
VideoFrame::AllocationSize(VideoFrame::I420, capture_format_.frame_size);
fake_frame_.reset(new uint8[fake_frame_size]);
}
void FakeVideoCaptureDevice::PopulateFormatRoster() {
DCHECK_EQ(capture_thread_.message_loop(), base::MessageLoop::current());
GetDeviceSupportedFormats(Name(), &format_roster_);
format_roster_index_ = 0;
}
}