blob: 65f663579935982271a0b8040ac5b03cb80710a7 [file] [log] [blame]
// Copyright 2017 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 "components/printing/browser/print_composite_client.h"
#include <utility>
#include "base/bind.h"
#include "base/memory/shared_memory_handle.h"
#include "base/metrics/histogram_macros.h"
#include "base/stl_util.h"
#include "components/printing/common/print_messages.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/render_frame_host.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/common/service_manager_connection.h"
#include "mojo/public/cpp/system/platform_handle.h"
#include "printing/printing_utils.h"
#include "services/service_manager/public/cpp/connector.h"
DEFINE_WEB_CONTENTS_USER_DATA_KEY(printing::PrintCompositeClient);
namespace {
uint64_t GenerateFrameGuid(int process_id, int frame_id) {
return static_cast<uint64_t>(process_id) << 32 | frame_id;
}
} // namespace
namespace printing {
PrintCompositeClient::PrintCompositeClient(content::WebContents* web_contents)
: content::WebContentsObserver(web_contents) {}
PrintCompositeClient::~PrintCompositeClient() {}
bool PrintCompositeClient::OnMessageReceived(
const IPC::Message& message,
content::RenderFrameHost* render_frame_host) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP_WITH_PARAM(PrintCompositeClient, message,
render_frame_host)
IPC_MESSAGE_HANDLER(PrintHostMsg_DidPrintFrameContent,
OnDidPrintFrameContent)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
void PrintCompositeClient::RenderFrameDeleted(
content::RenderFrameHost* render_frame_host) {
auto frame_guid = GenerateFrameGuid(render_frame_host->GetProcess()->GetID(),
render_frame_host->GetRoutingID());
auto iter = pending_subframe_cookies_.find(frame_guid);
if (iter != pending_subframe_cookies_.end()) {
// When a subframe we are expecting is deleted, we should notify pdf
// compositor service.
for (auto doc_cookie : iter->second) {
auto& compositor = GetCompositeRequest(doc_cookie);
compositor->NotifyUnavailableSubframe(frame_guid);
}
pending_subframe_cookies_.erase(iter);
}
}
void PrintCompositeClient::OnDidPrintFrameContent(
content::RenderFrameHost* render_frame_host,
int document_cookie,
const PrintHostMsg_DidPrintContent_Params& params) {
auto* outer_contents = web_contents()->GetOuterWebContents();
if (outer_contents) {
// When the printed content belongs to an extension or app page, the print
// composition needs to be handled by its outer content.
// TODO(weili): so far, we don't have printable web contents nested in more
// than one level. In the future, especially after PDF plugin is moved to
// OOPIF-based webview, we should check whether we need to handle web
// contents nested in multiple layers.
auto* outer_client = PrintCompositeClient::FromWebContents(outer_contents);
DCHECK(outer_client);
outer_client->OnDidPrintFrameContent(render_frame_host, document_cookie,
params);
return;
}
// Content in |params| is sent from untrusted source; only minimal processing
// is done here. Most of it will be directly forwarded to pdf compositor
// service.
auto& compositor = GetCompositeRequest(document_cookie);
mojo::ScopedSharedBufferHandle buffer_handle = mojo::WrapSharedMemoryHandle(
params.metafile_data_handle, params.data_size,
mojo::UnwrappedSharedMemoryHandleProtection::kReadOnly);
auto frame_guid = GenerateFrameGuid(render_frame_host->GetProcess()->GetID(),
render_frame_host->GetRoutingID());
compositor->AddSubframeContent(
frame_guid, std::move(buffer_handle),
ConvertContentInfoMap(web_contents(), render_frame_host,
params.subframe_content_info));
// Update our internal states about this frame.
pending_subframe_cookies_[frame_guid].erase(document_cookie);
if (pending_subframe_cookies_[frame_guid].empty())
pending_subframe_cookies_.erase(frame_guid);
printed_subframes_[document_cookie].insert(frame_guid);
}
void PrintCompositeClient::PrintCrossProcessSubframe(
const gfx::Rect& rect,
int document_cookie,
content::RenderFrameHost* subframe_host) {
PrintMsg_PrintFrame_Params params;
params.printable_area = rect;
params.document_cookie = document_cookie;
uint64_t frame_guid = GenerateFrameGuid(subframe_host->GetProcess()->GetID(),
subframe_host->GetRoutingID());
if (subframe_host->IsRenderFrameLive()) {
auto subframe_iter = printed_subframes_.find(document_cookie);
if (subframe_iter != printed_subframes_.end() &&
base::ContainsKey(subframe_iter->second, frame_guid)) {
// If this frame is already printed, no need to print again.
return;
}
auto cookie_iter = pending_subframe_cookies_.find(frame_guid);
if (cookie_iter != pending_subframe_cookies_.end() &&
base::ContainsKey(cookie_iter->second, document_cookie)) {
// If this frame is being printed, no need to print again.
return;
}
// Send the request to the destination frame.
subframe_host->Send(
new PrintMsg_PrintFrameContent(subframe_host->GetRoutingID(), params));
pending_subframe_cookies_[frame_guid].insert(document_cookie);
} else {
// When the subframe is dead, no need to send message,
// just notify the service.
auto& compositor = GetCompositeRequest(document_cookie);
compositor->NotifyUnavailableSubframe(frame_guid);
}
}
void PrintCompositeClient::DoCompositePageToPdf(
int document_cookie,
content::RenderFrameHost* render_frame_host,
int page_num,
base::SharedMemoryHandle handle,
uint32_t data_size,
const ContentToProxyIdMap& subframe_content_info,
mojom::PdfCompositor::CompositePageToPdfCallback callback) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
auto& compositor = GetCompositeRequest(document_cookie);
DCHECK(data_size);
mojo::ScopedSharedBufferHandle buffer_handle = mojo::WrapSharedMemoryHandle(
handle, data_size,
mojo::UnwrappedSharedMemoryHandleProtection::kReadOnly);
// Since this class owns compositor, compositor will be gone when this class
// is destructed. Mojo won't call its callback in that case so it is safe to
// use unretained |this| pointer here.
compositor->CompositePageToPdf(
GenerateFrameGuid(render_frame_host->GetProcess()->GetID(),
render_frame_host->GetRoutingID()),
page_num, std::move(buffer_handle),
ConvertContentInfoMap(web_contents(), render_frame_host,
subframe_content_info),
base::BindOnce(&PrintCompositeClient::OnDidCompositePageToPdf,
base::Unretained(this), std::move(callback)));
}
void PrintCompositeClient::DoCompositeDocumentToPdf(
int document_cookie,
content::RenderFrameHost* render_frame_host,
base::SharedMemoryHandle handle,
uint32_t data_size,
const ContentToProxyIdMap& subframe_content_info,
mojom::PdfCompositor::CompositeDocumentToPdfCallback callback) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
auto& compositor = GetCompositeRequest(document_cookie);
DCHECK(data_size);
mojo::ScopedSharedBufferHandle buffer_handle = mojo::WrapSharedMemoryHandle(
handle, data_size,
mojo::UnwrappedSharedMemoryHandleProtection::kReadOnly);
// Since this class owns compositor, compositor will be gone when this class
// is destructed. Mojo won't call its callback in that case so it is safe to
// use unretained |this| pointer here.
compositor->CompositeDocumentToPdf(
GenerateFrameGuid(render_frame_host->GetProcess()->GetID(),
render_frame_host->GetRoutingID()),
std::move(buffer_handle),
ConvertContentInfoMap(web_contents(), render_frame_host,
subframe_content_info),
base::BindOnce(&PrintCompositeClient::OnDidCompositeDocumentToPdf,
base::Unretained(this), document_cookie,
std::move(callback)));
}
void PrintCompositeClient::OnDidCompositePageToPdf(
printing::mojom::PdfCompositor::CompositePageToPdfCallback callback,
printing::mojom::PdfCompositor::Status status,
mojo::ScopedSharedBufferHandle handle) {
// Due to https://crbug.com/742517, we can not add and use COUNT for enums in
// mojo.
UMA_HISTOGRAM_ENUMERATION(
"CompositePageToPdf.Status", status,
static_cast<int32_t>(
printing::mojom::PdfCompositor::Status::COMPOSTING_FAILURE) +
1);
std::move(callback).Run(status, std::move(handle));
}
void PrintCompositeClient::OnDidCompositeDocumentToPdf(
int document_cookie,
printing::mojom::PdfCompositor::CompositeDocumentToPdfCallback callback,
printing::mojom::PdfCompositor::Status status,
mojo::ScopedSharedBufferHandle handle) {
RemoveCompositeRequest(document_cookie);
// Clear all stored printed subframes.
printed_subframes_.erase(document_cookie);
// Due to https://crbug.com/742517, we can not add and use COUNT for enums in
// mojo.
UMA_HISTOGRAM_ENUMERATION(
"CompositeDocToPdf.Status", status,
static_cast<int32_t>(
printing::mojom::PdfCompositor::Status::COMPOSTING_FAILURE) +
1);
std::move(callback).Run(status, std::move(handle));
}
ContentToFrameMap PrintCompositeClient::ConvertContentInfoMap(
content::WebContents* web_contents,
content::RenderFrameHost* render_frame_host,
const ContentToProxyIdMap& content_proxy_map) {
ContentToFrameMap content_frame_map;
int process_id = render_frame_host->GetProcess()->GetID();
for (auto& entry : content_proxy_map) {
auto content_id = entry.first;
auto proxy_id = entry.second;
// Find the RenderFrameHost that the proxy id corresponds to.
content::RenderFrameHost* rfh =
content::RenderFrameHost::FromPlaceholderId(process_id, proxy_id);
if (!rfh) {
// If we could not find the corresponding RenderFrameHost,
// just skip it.
continue;
}
// Store this frame's global unique id into the map.
content_frame_map[content_id] =
GenerateFrameGuid(rfh->GetProcess()->GetID(), rfh->GetRoutingID());
}
return content_frame_map;
}
mojom::PdfCompositorPtr& PrintCompositeClient::GetCompositeRequest(int cookie) {
auto iter = compositor_map_.find(cookie);
if (iter != compositor_map_.end()) {
DCHECK(iter->second.is_bound());
return iter->second;
}
auto iterator =
compositor_map_.emplace(cookie, CreateCompositeRequest()).first;
return iterator->second;
}
void PrintCompositeClient::RemoveCompositeRequest(int cookie) {
size_t erased = compositor_map_.erase(cookie);
DCHECK_EQ(erased, 1u);
}
mojom::PdfCompositorPtr PrintCompositeClient::CreateCompositeRequest() {
if (!connector_) {
service_manager::mojom::ConnectorRequest connector_request;
connector_ = service_manager::Connector::Create(&connector_request);
content::ServiceManagerConnection::GetForProcess()
->GetConnector()
->BindConnectorRequest(std::move(connector_request));
}
mojom::PdfCompositorPtr compositor;
connector_->BindInterface(mojom::kServiceName, &compositor);
compositor->SetWebContentsURL(web_contents()->GetLastCommittedURL());
return compositor;
}
} // namespace printing