File: rendering_window_manager.cc

package info (click to toggle)
chromium 139.0.7258.127-1
  • links: PTS, VCS
  • area: main
  • in suites:
  • size: 6,122,068 kB
  • sloc: cpp: 35,100,771; ansic: 7,163,530; javascript: 4,103,002; python: 1,436,920; asm: 946,517; xml: 746,709; pascal: 187,653; perl: 88,691; sh: 88,436; objc: 79,953; sql: 51,488; cs: 44,583; fortran: 24,137; makefile: 22,147; tcl: 15,277; php: 13,980; yacc: 8,984; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (79 lines) | stat: -rw-r--r-- 2,645 bytes parent folder | download | duplicates (10)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "ui/gfx/win/rendering_window_manager.h"

#include "base/functional/bind.h"
#include "base/logging.h"
#include "base/no_destructor.h"
#include "base/task/single_thread_task_runner.h"

namespace gfx {

// static
RenderingWindowManager* RenderingWindowManager::GetInstance() {
  static base::NoDestructor<RenderingWindowManager> instance;
  return instance.get();
}

void RenderingWindowManager::RegisterParent(HWND parent) {
  DCHECK(task_runner_->BelongsToCurrentThread());
  registered_hwnds_.emplace(parent, nullptr);
}

void RenderingWindowManager::RegisterChild(HWND parent,
                                           HWND child,
                                           DWORD expected_child_process_id) {
  if (!child)
    return;

  // This can be called from any thread, if we're not on the correct thread then
  // PostTask back to the UI thread before doing anything.
  if (!task_runner_->BelongsToCurrentThread()) {
    task_runner_->PostTask(
        FROM_HERE, base::BindOnce(&RenderingWindowManager::RegisterChild,
                                  base::Unretained(this), parent, child,
                                  expected_child_process_id));
    return;
  }

  // Check that |parent| was registered as a HWND that could have a child HWND.
  auto it = registered_hwnds_.find(parent);
  if (it == registered_hwnds_.end())
    return;

  // Check that |child| belongs to the GPU process.
  DWORD child_process_id = 0;
  DWORD child_thread_id = GetWindowThreadProcessId(child, &child_process_id);
  if (!child_thread_id || child_process_id != expected_child_process_id) {
    DLOG(ERROR) << "Child HWND not owned by GPU process.";
    return;
  }

  it->second = child;

  ::SetParent(child, parent);
  // Move D3D window behind Chrome's window to avoid losing some messages.
  ::SetWindowPos(child, HWND_BOTTOM, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE);
}

void RenderingWindowManager::UnregisterParent(HWND parent) {
  DCHECK(task_runner_->BelongsToCurrentThread());
  registered_hwnds_.erase(parent);
}

bool RenderingWindowManager::HasValidChildWindow(HWND parent) {
  DCHECK(task_runner_->BelongsToCurrentThread());
  auto it = registered_hwnds_.find(parent);
  if (it == registered_hwnds_.end())
    return false;
  return !!it->second && ::IsWindow(it->second);
}

RenderingWindowManager::RenderingWindowManager()
    : task_runner_(base::SingleThreadTaskRunner::GetCurrentDefault()) {}

RenderingWindowManager::~RenderingWindowManager() = default;

}  // namespace gfx