image_transport_factory_android.cc revision a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7
15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/browser/renderer_host/image_transport_factory_android.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
77dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch#include "base/lazy_instance.h"
87d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)#include "base/strings/stringprintf.h"
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/browser/gpu/browser_gpu_channel_host_factory.h"
105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/common/gpu/client/gl_helper.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "content/common/gpu/client/webgraphicscontext3d_command_buffer_impl.h"
1290dce4d38c5ff5333bea97d859d4e484e27edf0cTorne (Richard Coles)#include "content/common/gpu/gpu_process_launch_causes.h"
13868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)#include "third_party/WebKit/public/platform/WebGraphicsContext3D.h"
142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "third_party/khronos/GLES2/gl2.h"
15eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch#include "ui/gfx/android/device_display_info.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace content {
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
197dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochbase::LazyInstance<ObserverList<ImageTransportFactoryAndroidObserver> >::Leaky
207dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    g_factory_observers = LAZY_INSTANCE_INITIALIZER;
217dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
227dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochclass GLContextLostListener
23f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)    : public blink::WebGraphicsContext3D::WebGraphicsContextLostCallback {
247dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch public:
257dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // WebGraphicsContextLostCallback implementation.
267dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  virtual void onContextLost() OVERRIDE;
277dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch private:
287dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  static void DidLoseContext();
297dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch};
307dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)namespace {
322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)static ImageTransportFactoryAndroid* g_factory = NULL;
342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)class CmdBufferImageTransportFactory : public ImageTransportFactoryAndroid {
362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) public:
372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  CmdBufferImageTransportFactory();
382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual ~CmdBufferImageTransportFactory();
392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual uint32_t InsertSyncPoint() OVERRIDE;
412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual void WaitSyncPoint(uint32_t sync_point) OVERRIDE;
422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual uint32_t CreateTexture() OVERRIDE;
432a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual void DeleteTexture(uint32_t id) OVERRIDE;
442a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual void AcquireTexture(
452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)      uint32 texture_id, const signed char* mailbox_name) OVERRIDE;
46f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  virtual blink::WebGraphicsContext3D* GetContext3D() OVERRIDE {
472a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return context_.get();
482a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
492a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  virtual GLHelper* GetGLHelper() OVERRIDE;
50f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  virtual uint32 GetChannelID() OVERRIDE {
51a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7Torne (Richard Coles)    return BrowserGpuChannelHostFactory::instance()->GetGpuChannelId();
52f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  }
532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles) private:
552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  scoped_ptr<WebGraphicsContext3DCommandBufferImpl> context_;
562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  scoped_ptr<GLHelper> gl_helper_;
572a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
582a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  DISALLOW_COPY_AND_ASSIGN(CmdBufferImageTransportFactory);
592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)};
602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)CmdBufferImageTransportFactory::CmdBufferImageTransportFactory() {
621e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  BrowserGpuChannelHostFactory* factory =
631e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)      BrowserGpuChannelHostFactory::instance();
641e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  scoped_refptr<GpuChannelHost> gpu_channel_host(factory->EstablishGpuChannelSync(
651e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)      CAUSE_FOR_GPU_LAUNCH_WEBGRAPHICSCONTEXT3DCOMMANDBUFFERIMPL_INITIALIZE));
661e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  DCHECK(gpu_channel_host);
671e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)
68f2477e01787aa58f445919b809d89e252beef54fTorne (Richard Coles)  blink::WebGraphicsContext3D::Attributes attrs;
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  attrs.shareResources = true;
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  GURL url("chrome://gpu/ImageTransportFactoryAndroid");
71eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  static const size_t kBytesPerPixel = 4;
72eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch  gfx::DeviceDisplayInfo display_info;
731e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  size_t full_screen_texture_size_in_bytes = display_info.GetDisplayHeight() *
741e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                             display_info.GetDisplayWidth() *
751e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                             kBytesPerPixel;
761e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  WebGraphicsContext3DCommandBufferImpl::SharedMemoryLimits limits;
771e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  limits.command_buffer_size = 64 * 1024;
781e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  limits.start_transfer_buffer_size = 64 * 1024;
791e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  limits.min_transfer_buffer_size = 64 * 1024;
801e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  limits.max_transfer_buffer_size = std::min(
811e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)      3 * full_screen_texture_size_in_bytes, kDefaultMaxTransferBufferSize);
821e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  limits.mapped_memory_reclaim_limit =
831e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)      WebGraphicsContext3DCommandBufferImpl::kNoLimit;
841e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)  context_.reset(
851e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)      new WebGraphicsContext3DCommandBufferImpl(0,  // offscreen
861e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                                url,
871e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                                gpu_channel_host.get(),
881e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                                attrs,
891e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                                false,
901e9bf3e0803691d0a228da41fc608347b6db4340Torne (Richard Coles)                                                limits));
917dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  context_->setContextLostCallback(context_lost_listener_.get());
927d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)  if (context_->makeContextCurrent())
937d4cd473f85ac64c3747c96c277f9e506a0d2246Torne (Richard Coles)    context_->pushGroupMarkerEXT(
94eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch        base::StringPrintf("CmdBufferImageTransportFactory-%p",
95eb525c5499e34cc9c4b825d6d9e75bb07cc06aceBen Murdoch                           context_.get()).c_str());
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
982a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)CmdBufferImageTransportFactory::~CmdBufferImageTransportFactory() {
997dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  context_->setContextLostCallback(NULL);
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1022a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)uint32_t CmdBufferImageTransportFactory::InsertSyncPoint() {
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!context_->makeContextCurrent()) {
1042a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    LOG(ERROR) << "Failed to make helper context current.";
1052a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return 0;
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1072a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return context_->insertSyncPoint();
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1102a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void CmdBufferImageTransportFactory::WaitSyncPoint(uint32_t sync_point) {
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!context_->makeContextCurrent()) {
1122a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    LOG(ERROR) << "Failed to make helper context current.";
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1152a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  context_->waitSyncPoint(sync_point);
1162a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1182a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)uint32_t CmdBufferImageTransportFactory::CreateTexture() {
1192a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (!context_->makeContextCurrent()) {
1202a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    LOG(ERROR) << "Failed to make helper context current.";
1212a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return false;
1222a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1232a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return context_->createTexture();
1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1262a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void CmdBufferImageTransportFactory::DeleteTexture(uint32_t id) {
1272a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (!context_->makeContextCurrent()) {
1282a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    LOG(ERROR) << "Failed to make helper context current.";
1292a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return;
1302a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1312a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  context_->deleteTexture(id);
1322a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1332a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1342a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)void CmdBufferImageTransportFactory::AcquireTexture(
1352a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    uint32 texture_id, const signed char* mailbox_name) {
1362a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  if (!context_->makeContextCurrent()) {
1372a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    LOG(ERROR) << "Failed to make helper context current.";
1382a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)    return;
1392a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  }
1402a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  context_->bindTexture(GL_TEXTURE_2D, texture_id);
1412a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  context_->consumeTextureCHROMIUM(GL_TEXTURE_2D, mailbox_name);
1422a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  context_->flush();
1435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1445821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1452a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)GLHelper* CmdBufferImageTransportFactory::GetGLHelper() {
146c2e0dbddbe15c98d52c4786dac06cb8952a8ae6dTorne (Richard Coles)  if (!gl_helper_)
1478bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)    gl_helper_.reset(new GLHelper(context_.get(),
1488bcbed890bc3ce4d7a057a8f32cab53fa534672eTorne (Richard Coles)                                  context_->GetContextSupport()));
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return gl_helper_.get();
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1532a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}  // anonymous namespace
1542a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1552a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)// static
1562a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)ImageTransportFactoryAndroid* ImageTransportFactoryAndroid::GetInstance() {
1574e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)  if (!g_factory)
1584e180b6a0b4720a9b8e9e959a882386f690f08ffTorne (Richard Coles)    g_factory = new CmdBufferImageTransportFactory();
1592a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1602a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)  return g_factory;
1612a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1622a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1637dbb3d5cf0c15f500944d211057644d6a2f37371Ben MurdochImageTransportFactoryAndroid::ImageTransportFactoryAndroid()
1647dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    : context_lost_listener_(new GLContextLostListener()) {}
1657dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
1667dbb3d5cf0c15f500944d211057644d6a2f37371Ben MurdochImageTransportFactoryAndroid::~ImageTransportFactoryAndroid() {}
1677dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
1687dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochvoid ImageTransportFactoryAndroid::AddObserver(
1697dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    ImageTransportFactoryAndroidObserver* observer) {
1707dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  g_factory_observers.Get().AddObserver(observer);
1717dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch}
1727dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
1737dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochvoid ImageTransportFactoryAndroid::RemoveObserver(
1747dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch    ImageTransportFactoryAndroidObserver* observer) {
1757dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  g_factory_observers.Get().RemoveObserver(observer);
1767dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch}
1777dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch
1787dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochvoid GLContextLostListener::onContextLost() {
1797dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // Need to post a task because the command buffer client cannot be deleted
1807dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  // from within this callback.
1817dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  LOG(ERROR) << "Context lost.";
1827dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  base::MessageLoop::current()->PostTask(
1837dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      FROM_HERE,
1847dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      base::Bind(&GLContextLostListener::DidLoseContext));
1852a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1862a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1877dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdochvoid GLContextLostListener::DidLoseContext() {
1887dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  delete g_factory;
1897dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  g_factory = NULL;
1907dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch  FOR_EACH_OBSERVER(ImageTransportFactoryAndroidObserver,
1917dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      g_factory_observers.Get(),
1927dbb3d5cf0c15f500944d211057644d6a2f37371Ben Murdoch      OnLostResources());
1932a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)}
1942a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)
1952a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)} // namespace content
196