// Copyright (c) 2012 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 "remoting/codec/video_encoder_verbatim.h" #include "base/logging.h" #include "base/stl_util.h" #include "base/time/time.h" #include "remoting/base/util.h" #include "remoting/proto/video.pb.h" #include "third_party/webrtc/modules/desktop_capture/desktop_frame.h" namespace remoting { VideoEncoderVerbatim::VideoEncoderVerbatim() {} VideoEncoderVerbatim::~VideoEncoderVerbatim() {} scoped_ptr VideoEncoderVerbatim::Encode( const webrtc::DesktopFrame& frame) { CHECK(frame.data()); base::Time encode_start_time = base::Time::Now(); scoped_ptr packet(new VideoPacket()); VideoPacketFormat* format = packet->mutable_format(); format->set_encoding(VideoPacketFormat::ENCODING_VERBATIM); if (!frame.size().equals(screen_size_)) { screen_size_ = frame.size(); format->set_screen_width(screen_size_.width()); format->set_screen_height(screen_size_.height()); } // Calculate output size. size_t output_size = 0; for (webrtc::DesktopRegion::Iterator iter(frame.updated_region()); !iter.IsAtEnd(); iter.Advance()) { const webrtc::DesktopRect& rect = iter.rect(); output_size += rect.width() * rect.height() * webrtc::DesktopFrame::kBytesPerPixel; } uint8_t* out = GetOutputBuffer(packet.get(), output_size); const int in_stride = frame.stride(); // Store all changed rectangles in the packet. for (webrtc::DesktopRegion::Iterator iter(frame.updated_region()); !iter.IsAtEnd(); iter.Advance()) { const webrtc::DesktopRect& rect = iter.rect(); const int row_size = webrtc::DesktopFrame::kBytesPerPixel * rect.width(); const uint8_t* in = frame.data() + rect.top() * in_stride + rect.left() * webrtc::DesktopFrame::kBytesPerPixel; for (int y = rect.top(); y < rect.top() + rect.height(); ++y) { memcpy(out, in, row_size); out += row_size; in += in_stride; } Rect* dirty_rect = packet->add_dirty_rects(); dirty_rect->set_x(rect.left()); dirty_rect->set_y(rect.top()); dirty_rect->set_width(rect.width()); dirty_rect->set_height(rect.height()); } packet->set_capture_time_ms(frame.capture_time_ms()); packet->set_encode_time_ms( (base::Time::Now() - encode_start_time).InMillisecondsRoundedUp()); if (!frame.dpi().is_zero()) { packet->mutable_format()->set_x_dpi(frame.dpi().x()); packet->mutable_format()->set_y_dpi(frame.dpi().y()); } return packet.Pass(); } uint8_t* VideoEncoderVerbatim::GetOutputBuffer(VideoPacket* packet, size_t size) { packet->mutable_data()->resize(size); return reinterpret_cast(string_as_array(packet->mutable_data())); } } // namespace remoting