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
base / win / com_init_balancer.cc [blame]
// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/win/com_init_balancer.h"
#include <objbase.h>
#include "base/check_op.h"
namespace base {
namespace win {
namespace internal {
ComInitBalancer::ComInitBalancer(DWORD co_init) : co_init_(co_init) {
ULARGE_INTEGER spy_cookie = {};
HRESULT hr = ::CoRegisterInitializeSpy(this, &spy_cookie);
if (SUCCEEDED(hr))
spy_cookie_ = spy_cookie;
}
ComInitBalancer::~ComInitBalancer() {
DCHECK(!spy_cookie_.has_value());
}
void ComInitBalancer::Disable() {
if (spy_cookie_.has_value()) {
::CoRevokeInitializeSpy(spy_cookie_.value());
reference_count_ = 0;
spy_cookie_.reset();
}
}
DWORD ComInitBalancer::GetReferenceCountForTesting() const {
return reference_count_;
}
IFACEMETHODIMP
ComInitBalancer::PreInitialize(DWORD apartment_type, DWORD reference_count) {
return S_OK;
}
IFACEMETHODIMP
ComInitBalancer::PostInitialize(HRESULT result,
DWORD apartment_type,
DWORD new_reference_count) {
reference_count_ = new_reference_count;
return result;
}
IFACEMETHODIMP
ComInitBalancer::PreUninitialize(DWORD reference_count) {
if (reference_count == 1 && spy_cookie_.has_value()) {
// Increase the reference count to prevent premature and unbalanced
// uninitalization of the COM library.
::CoInitializeEx(nullptr, co_init_);
}
return S_OK;
}
IFACEMETHODIMP
ComInitBalancer::PostUninitialize(DWORD new_reference_count) {
reference_count_ = new_reference_count;
return S_OK;
}
} // namespace internal
} // namespace win
} // namespace base