| [email protected] | fd911dd | 2012-01-27 01:57:10 | [diff] [blame] | 1 | // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| license.bot | bf09a50 | 2008-08-24 00:55:55 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| initial.commit | 09911bf | 2008-07-26 23:55:29 | [diff] [blame] | 4 | |
| [email protected] | d353541f | 2012-05-03 22:45:41 | [diff] [blame] | 5 | #include "content/renderer/render_process_impl.h" |
| 6 | |
| [email protected] | 037fce0 | 2009-01-22 01:42:15 | [diff] [blame] | 7 | #include "build/build_config.h" |
| 8 | |
| [email protected] | 037fce0 | 2009-01-22 01:42:15 | [diff] [blame] | 9 | #if defined(OS_WIN) |
| initial.commit | 09911bf | 2008-07-26 23:55:29 | [diff] [blame] | 10 | #include <windows.h> |
| 11 | #include <objidl.h> |
| 12 | #include <mlang.h> |
| [email protected] | 037fce0 | 2009-01-22 01:42:15 | [diff] [blame] | 13 | #endif |
| initial.commit | 09911bf | 2008-07-26 23:55:29 | [diff] [blame] | 14 | |
| fdoray | d2233a7 | 2016-12-13 17:18:21 | [diff] [blame] | 15 | #include <stddef.h> |
| 16 | |
| fdoray | 743f8d6d | 2017-02-09 15:24:23 | [diff] [blame] | 17 | #include <algorithm> |
| fdoray | 31cc6f8 | 2017-02-10 23:31:10 | [diff] [blame] | 18 | #include <utility> |
| fdoray | d2233a7 | 2016-12-13 17:18:21 | [diff] [blame] | 19 | |
| Eric Holk | 1384f6d | 2018-01-05 00:49:36 | [diff] [blame] | 20 | #include "base/base_switches.h" |
| fdoray | d2233a7 | 2016-12-13 17:18:21 | [diff] [blame] | 21 | #include "base/bind.h" |
| initial.commit | 09911bf | 2008-07-26 23:55:29 | [diff] [blame] | 22 | #include "base/command_line.h" |
| [email protected] | 037fce0 | 2009-01-22 01:42:15 | [diff] [blame] | 23 | #include "base/compiler_specific.h" |
| georgesak | 80353b5 | 2017-01-10 21:18:51 | [diff] [blame] | 24 | #include "base/debug/crash_logging.h" |
| Eric Holk | dc499db | 2017-07-17 17:57:35 | [diff] [blame] | 25 | #include "base/debug/stack_trace.h" |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 26 | #include "base/feature_list.h" |
| fdoray | 31cc6f8 | 2017-02-10 23:31:10 | [diff] [blame] | 27 | #include "base/memory/ptr_util.h" |
| Ross McIlroy | 900375b | 2019-05-16 20:17:42 | [diff] [blame] | 28 | #include "base/strings/string_split.h" |
| Sebastien Marchand | 75a7cdf | 2018-11-13 23:47:03 | [diff] [blame] | 29 | #include "base/system/sys_info.h" |
| Gabriel Charette | 52fa3ae | 2019-04-15 21:44:37 | [diff] [blame] | 30 | #include "base/task/thread_pool/initialization_util.h" |
| Francois Doray | 7f77731 | 2019-05-16 12:26:31 | [diff] [blame] | 31 | #include "base/task/thread_pool/thread_pool.h" |
| fdoray | d2233a7 | 2016-12-13 17:18:21 | [diff] [blame] | 32 | #include "base/time/time.h" |
| Gabriel Charette | 52fa3ae | 2019-04-15 21:44:37 | [diff] [blame] | 33 | #include "content/common/thread_pool_util.h" |
| sammc | 7f6c6a0 | 2017-01-30 00:53:51 | [diff] [blame] | 34 | #include "content/public/common/bindings_policy.h" |
| fdoray | d2233a7 | 2016-12-13 17:18:21 | [diff] [blame] | 35 | #include "content/public/common/content_client.h" |
| bradnelson | c79f5a6f | 2016-10-10 18:31:14 | [diff] [blame] | 36 | #include "content/public/common/content_features.h" |
| [email protected] | c08950d2 | 2011-10-13 22:20:29 | [diff] [blame] | 37 | #include "content/public/common/content_switches.h" |
| [email protected] | d344114c | 2011-10-01 01:24:34 | [diff] [blame] | 38 | #include "content/public/renderer/content_renderer_client.h" |
| Eric Holk | 1384f6d | 2018-01-05 00:49:36 | [diff] [blame] | 39 | #include "services/service_manager/embedder/switches.h" |
| Blink Reformat | a30d423 | 2018-04-07 15:31:06 | [diff] [blame] | 40 | #include "third_party/blink/public/web/web_frame.h" |
| [email protected] | 067f519 | 2014-01-29 05:22:09 | [diff] [blame] | 41 | #include "v8/include/v8.h" |
| initial.commit | 09911bf | 2008-07-26 23:55:29 | [diff] [blame] | 42 | |
| georgesak | 80353b5 | 2017-01-10 21:18:51 | [diff] [blame] | 43 | #if defined(OS_WIN) |
| 44 | #include "base/win/win_util.h" |
| 45 | #endif |
| 46 | |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 47 | namespace { |
| 48 | |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 49 | void SetV8FlagIfFeature(const base::Feature& feature, const char* v8_flag) { |
| 50 | if (base::FeatureList::IsEnabled(feature)) { |
| 51 | v8::V8::SetFlagsFromString(v8_flag, strlen(v8_flag)); |
| 52 | } |
| 53 | } |
| 54 | |
| bradnelson | 2730e351 | 2017-01-21 20:32:21 | [diff] [blame] | 55 | void SetV8FlagIfNotFeature(const base::Feature& feature, const char* v8_flag) { |
| 56 | if (!base::FeatureList::IsEnabled(feature)) { |
| 57 | v8::V8::SetFlagsFromString(v8_flag, strlen(v8_flag)); |
| 58 | } |
| 59 | } |
| 60 | |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 61 | void SetV8FlagIfHasSwitch(const char* switch_name, const char* v8_flag) { |
| 62 | if (base::CommandLine::ForCurrentProcess()->HasSwitch(switch_name)) { |
| 63 | v8::V8::SetFlagsFromString(v8_flag, strlen(v8_flag)); |
| 64 | } |
| 65 | } |
| 66 | |
| Gabriel Charette | 43fd370 | 2019-05-29 16:36:51 | [diff] [blame^] | 67 | std::unique_ptr<base::ThreadPoolInstance::InitParams> |
| 68 | GetThreadPoolInitParams() { |
| Etienne Pierre-doray | ce56296 | 2019-02-08 18:50:48 | [diff] [blame] | 69 | constexpr int kMaxNumThreadsInForegroundPoolLowerBound = 3; |
| Gabriel Charette | 43fd370 | 2019-05-29 16:36:51 | [diff] [blame^] | 70 | return std::make_unique<base::ThreadPoolInstance::InitParams>( |
| Francois Doray | 7f77731 | 2019-05-16 12:26:31 | [diff] [blame] | 71 | std::max(kMaxNumThreadsInForegroundPoolLowerBound, |
| 72 | content::GetMinForegroundThreadsInRendererThreadPool())); |
| fdoray | d2233a7 | 2016-12-13 17:18:21 | [diff] [blame] | 73 | } |
| 74 | |
| Tomas Popela | afffa97 | 2018-11-13 20:42:05 | [diff] [blame] | 75 | #if defined(DCHECK_IS_CONFIGURABLE) |
| Sigurdur Asgeirsson | 379c51e4 | 2017-09-21 12:52:45 | [diff] [blame] | 76 | void V8DcheckCallbackHandler(const char* file, int line, const char* message) { |
| 77 | // TODO(siggi): Set a crash key or a breadcrumb so the fact that we hit a |
| 78 | // V8 DCHECK gets out in the crash report. |
| 79 | ::logging::LogMessage(file, line, logging::LOG_DCHECK).stream() << message; |
| 80 | } |
| Tomas Popela | afffa97 | 2018-11-13 20:42:05 | [diff] [blame] | 81 | #endif // defined(DCHECK_IS_CONFIGURABLE) |
| Sigurdur Asgeirsson | 379c51e4 | 2017-09-21 12:52:45 | [diff] [blame] | 82 | |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 83 | } // namespace |
| 84 | |
| [email protected] | eb39819 | 2012-10-22 20:16:19 | [diff] [blame] | 85 | namespace content { |
| 86 | |
| Francois Doray | 7f77731 | 2019-05-16 12:26:31 | [diff] [blame] | 87 | RenderProcessImpl::RenderProcessImpl() |
| 88 | : RenderProcess("Renderer", GetThreadPoolInitParams()), |
| fdoray | 31cc6f8 | 2017-02-10 23:31:10 | [diff] [blame] | 89 | enabled_bindings_(0) { |
| Tomas Popela | afffa97 | 2018-11-13 20:42:05 | [diff] [blame] | 90 | #if defined(DCHECK_IS_CONFIGURABLE) |
| Wez | a6ca5b9 | 2018-03-23 19:03:07 | [diff] [blame] | 91 | // Some official builds ship with DCHECKs compiled in. Failing DCHECKs then |
| 92 | // are either fatal or simply log the error, based on a feature flag. |
| Sigurdur Asgeirsson | 379c51e4 | 2017-09-21 12:52:45 | [diff] [blame] | 93 | // Make sure V8 follows suit by setting a Dcheck handler that forwards to |
| 94 | // the Chrome base logging implementation. |
| 95 | v8::V8::SetDcheckErrorHandler(&V8DcheckCallbackHandler); |
| 96 | |
| Wez | a6ca5b9 | 2018-03-23 19:03:07 | [diff] [blame] | 97 | if (!base::FeatureList::IsEnabled(base::kDCheckIsFatalFeature)) { |
| Sigurdur Asgeirsson | 379c51e4 | 2017-09-21 12:52:45 | [diff] [blame] | 98 | // These V8 flags default on in this build configuration. This triggers |
| 99 | // additional verification and code generation, which both slows down V8, |
| 100 | // and can lead to fatal CHECKs. Turn these flags down to get something |
| 101 | // closer to V8s normal performance and behavior. |
| 102 | constexpr char kDisabledFlags[] = |
| 103 | "--noturbo_verify " |
| Sigurdur Asgeirsson | 379c51e4 | 2017-09-21 12:52:45 | [diff] [blame] | 104 | "--noturbo_verify_allocation " |
| 105 | "--nodebug_code"; |
| 106 | |
| 107 | v8::V8::SetFlagsFromString(kDisabledFlags, sizeof(kDisabledFlags)); |
| 108 | } |
| Tomas Popela | afffa97 | 2018-11-13 20:42:05 | [diff] [blame] | 109 | #endif // defined(DCHECK_IS_CONFIGURABLE) |
| Sigurdur Asgeirsson | 379c51e4 | 2017-09-21 12:52:45 | [diff] [blame] | 110 | |
| [email protected] | 35b4f0c | 2014-06-26 16:55:27 | [diff] [blame] | 111 | if (base::SysInfo::IsLowEndDevice()) { |
| [email protected] | 067f519 | 2014-01-29 05:22:09 | [diff] [blame] | 112 | std::string optimize_flag("--optimize-for-size"); |
| Clemens Hammacher | c4a139a | 2019-04-25 13:55:09 | [diff] [blame] | 113 | v8::V8::SetFlagsFromString(optimize_flag.c_str(), optimize_flag.size()); |
| [email protected] | 067f519 | 2014-01-29 05:22:09 | [diff] [blame] | 114 | } |
| [email protected] | 987422f | 2013-10-01 10:33:31 | [diff] [blame] | 115 | |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 116 | SetV8FlagIfHasSwitch(switches::kDisableJavaScriptHarmonyShipping, |
| 117 | "--noharmony-shipping"); |
| 118 | SetV8FlagIfHasSwitch(switches::kJavaScriptHarmony, "--harmony"); |
| Kouhei Ueno | 14d350d | 2018-10-01 02:36:27 | [diff] [blame] | 119 | |
| 120 | constexpr char kModuleFlags[] = |
| 121 | "--harmony-dynamic-import --harmony-import-meta"; |
| 122 | v8::V8::SetFlagsFromString(kModuleFlags, sizeof(kModuleFlags)); |
| 123 | |
| Michael Hablich | 896d5266 | 2017-10-23 15:59:57 | [diff] [blame] | 124 | SetV8FlagIfFeature(features::kV8VmFuture, "--future"); |
| 125 | SetV8FlagIfNotFeature(features::kV8VmFuture, "--no-future"); |
| Clemens Hammacher | 0c8a15a | 2018-04-27 13:45:32 | [diff] [blame] | 126 | |
| Clemens Hammacher | fab5510 | 2018-07-24 16:19:19 | [diff] [blame] | 127 | SetV8FlagIfFeature(features::kWebAssemblyBaseline, |
| 128 | "--liftoff --wasm-tier-up"); |
| 129 | SetV8FlagIfNotFeature(features::kWebAssemblyBaseline, |
| 130 | "--no-liftoff --no-wasm-tier-up"); |
| Clemens Hammacher | 0c8a15a | 2018-04-27 13:45:32 | [diff] [blame] | 131 | |
| Clemens Hammacher | 082fa57 | 2019-04-25 08:22:29 | [diff] [blame] | 132 | SetV8FlagIfFeature(features::kWebAssemblyCodeGC, "--wasm-code-gc"); |
| 133 | SetV8FlagIfNotFeature(features::kWebAssemblyCodeGC, "--no-wasm-code-gc"); |
| 134 | |
| Deepti Gandluri | 11734cc4 | 2019-05-02 18:00:20 | [diff] [blame] | 135 | SetV8FlagIfFeature(features::kWebAssemblySimd, "--experimental-wasm-simd"); |
| 136 | SetV8FlagIfNotFeature(features::kWebAssemblySimd, |
| 137 | "--no-experimental-wasm-simd"); |
| 138 | |
| Ben Smith | 24c1e5c | 2018-06-20 01:09:02 | [diff] [blame] | 139 | if (base::FeatureList::IsEnabled(features::kWebAssemblyThreads)) { |
| 140 | constexpr char kFlags[] = |
| 141 | "--harmony-sharedarraybuffer " |
| 142 | "--no-wasm-disable-structured-cloning " |
| 143 | "--experimental-wasm-threads"; |
| 144 | |
| 145 | v8::V8::SetFlagsFromString(kFlags, sizeof(kFlags)); |
| 146 | } else { |
| 147 | SetV8FlagIfNotFeature(features::kWebAssembly, |
| 148 | "--wasm-disable-structured-cloning"); |
| 149 | SetV8FlagIfFeature(features::kSharedArrayBuffer, |
| 150 | "--harmony-sharedarraybuffer"); |
| 151 | SetV8FlagIfNotFeature(features::kSharedArrayBuffer, |
| 152 | "--no-harmony-sharedarraybuffer"); |
| 153 | } |
| Michael Hablich | 896d5266 | 2017-10-23 15:59:57 | [diff] [blame] | 154 | |
| Michael Lippautz | e0fb49a3 | 2019-05-21 14:11:44 | [diff] [blame] | 155 | SetV8FlagIfFeature(features::kBlinkHeapUnifiedGCScheduling, |
| 156 | "--global-gc-scheduling"); |
| 157 | |
| Eric Holk | b4f6013 | 2017-08-18 19:37:41 | [diff] [blame] | 158 | SetV8FlagIfNotFeature(features::kWebAssemblyTrapHandler, |
| 159 | "--no-wasm-trap-handler"); |
| Lei Zhang | 2d3ead60 | 2018-07-10 01:15:50 | [diff] [blame] | 160 | #if defined(OS_LINUX) && defined(ARCH_CPU_X86_64) |
| Eric Holk | dc499db | 2017-07-17 17:57:35 | [diff] [blame] | 161 | if (base::FeatureList::IsEnabled(features::kWebAssemblyTrapHandler)) { |
| Eric Holk | 1384f6d | 2018-01-05 00:49:36 | [diff] [blame] | 162 | base::CommandLine* command_line = base::CommandLine::ForCurrentProcess(); |
| 163 | if (!command_line->HasSwitch( |
| 164 | service_manager::switches::kDisableInProcessStackTraces)) { |
| Andreas Haas | ef19d59 | 2019-04-30 18:16:51 | [diff] [blame] | 165 | // Only enable WebAssembly trap handler if we can set the callback. |
| 166 | if (base::debug::SetStackDumpFirstChanceCallback( |
| 167 | v8::V8::TryHandleSignal)) { |
| 168 | // We registered the WebAssembly trap handler callback with the stack |
| 169 | // dump signal handler successfully. We can tell V8 that it can enable |
| 170 | // WebAssembly trap handler without using the V8 signal handler. |
| 171 | v8::V8::EnableWebAssemblyTrapHandler(/*use_v8_signal_handler=*/false); |
| 172 | } |
| Eric Holk | 1384f6d | 2018-01-05 00:49:36 | [diff] [blame] | 173 | } else if (!command_line->HasSwitch(switches::kEnableCrashReporter) && |
| 174 | !command_line->HasSwitch( |
| 175 | switches::kEnableCrashReporterForTesting)) { |
| 176 | // If we are using WebAssembly trap handling but both Breakpad and |
| 177 | // in-process stack traces are disabled then there will be no signal |
| 178 | // handler. In this case, we fall back on V8's default handler |
| 179 | // (https://crbug.com/798150). |
| Andreas Haas | ef19d59 | 2019-04-30 18:16:51 | [diff] [blame] | 180 | v8::V8::EnableWebAssemblyTrapHandler(/*use_v8_signal_handler=*/true); |
| Eric Holk | 1384f6d | 2018-01-05 00:49:36 | [diff] [blame] | 181 | } |
| Eric Holk | dc499db | 2017-07-17 17:57:35 | [diff] [blame] | 182 | } |
| 183 | #endif |
| Andreas Haas | 5ed0f50 | 2018-11-06 09:05:06 | [diff] [blame] | 184 | #if defined(OS_WIN) && defined(ARCH_CPU_X86_64) |
| 185 | if (base::FeatureList::IsEnabled(features::kWebAssemblyTrapHandler)) { |
| 186 | // On Windows we use the default trap handler provided by V8. |
| 187 | bool use_v8_trap_handler = true; |
| 188 | v8::V8::EnableWebAssemblyTrapHandler(use_v8_trap_handler); |
| 189 | } |
| 190 | #endif |
| Andreas Haas | 3bd4532 | 2018-11-21 07:45:42 | [diff] [blame] | 191 | #if defined(OS_MACOSX) && defined(ARCH_CPU_X86_64) |
| 192 | if (base::FeatureList::IsEnabled(features::kWebAssemblyTrapHandler)) { |
| 193 | // On macOS, Crashpad uses exception ports to handle signals in a different |
| 194 | // process. As we cannot just pass a callback to this other process, we ask |
| 195 | // V8 to install its own signal handler to deal with WebAssembly traps. |
| 196 | bool use_v8_signal_handler = true; |
| 197 | v8::V8::EnableWebAssemblyTrapHandler(use_v8_signal_handler); |
| 198 | } |
| 199 | #endif // defined(OS_MACOSX) && defined(ARCH_CPU_X86_64) |
| Eric Holk | dc499db | 2017-07-17 17:57:35 | [diff] [blame] | 200 | |
| avi | 83883c8 | 2014-12-23 00:08:49 | [diff] [blame] | 201 | const base::CommandLine& command_line = |
| 202 | *base::CommandLine::ForCurrentProcess(); |
| ishell | 75fddc1 | 2016-04-12 14:03:14 | [diff] [blame] | 203 | |
| Ross McIlroy | 3ba9207 | 2018-08-01 00:43:30 | [diff] [blame] | 204 | if (command_line.HasSwitch(switches::kNoV8UntrustedCodeMitigations)) { |
| 205 | const char* disable_mitigations = "--no-untrusted-code-mitigations"; |
| 206 | v8::V8::SetFlagsFromString(disable_mitigations, |
| 207 | strlen(disable_mitigations)); |
| 208 | } |
| 209 | |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 210 | if (command_line.HasSwitch(switches::kJavaScriptFlags)) { |
| Ross McIlroy | 900375b | 2019-05-16 20:17:42 | [diff] [blame] | 211 | std::string js_flags = |
| 212 | command_line.GetSwitchValueASCII(switches::kJavaScriptFlags); |
| 213 | std::vector<base::StringPiece> flag_list = base::SplitStringPiece( |
| 214 | js_flags, ",", base::TRIM_WHITESPACE, base::SPLIT_WANT_NONEMPTY); |
| 215 | for (const auto& flag : flag_list) { |
| 216 | v8::V8::SetFlagsFromString(flag.as_string().c_str(), flag.size()); |
| 217 | } |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 218 | } |
| [email protected] | 55dd933 | 2013-09-04 17:17:50 | [diff] [blame] | 219 | |
| sammc | 7f6c6a0 | 2017-01-30 00:53:51 | [diff] [blame] | 220 | if (command_line.HasSwitch(switches::kDomAutomationController)) |
| 221 | enabled_bindings_ |= BINDINGS_POLICY_DOM_AUTOMATION; |
| 222 | if (command_line.HasSwitch(switches::kStatsCollectionController)) |
| 223 | enabled_bindings_ |= BINDINGS_POLICY_STATS_COLLECTION; |
| [email protected] | e68e62fa | 2009-02-20 02:00:04 | [diff] [blame] | 224 | } |
| 225 | |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 226 | RenderProcessImpl::~RenderProcessImpl() { |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 227 | #ifndef NDEBUG |
| Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 228 | int count = blink::WebFrame::InstanceCount(); |
| [email protected] | 6bd867b | 2013-07-24 22:10:20 | [diff] [blame] | 229 | if (count) |
| 230 | DLOG(ERROR) << "WebFrame LEAKED " << count << " TIMES"; |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 231 | #endif |
| [email protected] | e68e62fa | 2009-02-20 02:00:04 | [diff] [blame] | 232 | |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 233 | GetShutDownEvent()->Signal(); |
| [email protected] | 396c3a46 | 2010-03-03 05:03:22 | [diff] [blame] | 234 | } |
| [email protected] | e68e62fa | 2009-02-20 02:00:04 | [diff] [blame] | 235 | |
| fdoray | 31cc6f8 | 2017-02-10 23:31:10 | [diff] [blame] | 236 | std::unique_ptr<RenderProcess> RenderProcessImpl::Create() { |
| Francois Doray | 7f77731 | 2019-05-16 12:26:31 | [diff] [blame] | 237 | return base::WrapUnique(new RenderProcessImpl()); |
| fdoray | 31cc6f8 | 2017-02-10 23:31:10 | [diff] [blame] | 238 | } |
| 239 | |
| [email protected] | 744c2a2 | 2012-03-15 18:42:04 | [diff] [blame] | 240 | void RenderProcessImpl::AddBindings(int bindings) { |
| 241 | enabled_bindings_ |= bindings; |
| 242 | } |
| 243 | |
| 244 | int RenderProcessImpl::GetEnabledBindings() const { |
| 245 | return enabled_bindings_; |
| 246 | } |
| 247 | |
| Arthur Sonzogni | c4f8dee | 2018-09-05 08:51:33 | [diff] [blame] | 248 | void RenderProcessImpl::AddRefProcess() { |
| 249 | NOTREACHED(); |
| 250 | } |
| 251 | |
| 252 | void RenderProcessImpl::ReleaseProcess() { |
| 253 | NOTREACHED(); |
| 254 | } |
| 255 | |
| [email protected] | eb39819 | 2012-10-22 20:16:19 | [diff] [blame] | 256 | } // namespace content |