103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)// Copyright 2014 The Chromium Authors. All rights reserved.
203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)// found in the LICENSE file.
403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/examples/surfaces_app/child_gl_impl.h"
603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#ifndef GL_GLEXT_PROTOTYPES
803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#define GL_GLEXT_PROTOTYPES
903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#endif
1003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
1103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "base/bind.h"
1203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "base/message_loop/message_loop.h"
1303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "cc/output/compositor_frame.h"
1403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "cc/output/delegated_frame_data.h"
1503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "cc/quads/render_pass.h"
1603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "cc/quads/texture_draw_quad.h"
1703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "gpu/GLES2/gl2chromium.h"
1803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "gpu/GLES2/gl2extchromium.h"
1903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/examples/surfaces_app/surfaces_util.h"
2003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/public/cpp/application/application_connection.h"
2103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/public/cpp/environment/environment.h"
2203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/services/public/cpp/geometry/geometry_type_converters.h"
2303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/services/public/cpp/surfaces/surfaces_type_converters.h"
2403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/services/public/interfaces/surfaces/surface_id.mojom.h"
2503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "mojo/services/public/interfaces/surfaces/surfaces.mojom.h"
2603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "third_party/khronos/GLES2/gl2.h"
2703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "third_party/khronos/GLES2/gl2ext.h"
2803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "ui/gfx/rect.h"
2903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)#include "ui/gfx/transform.h"
3003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
3103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)namespace mojo {
3203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)namespace examples {
3303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
3403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)using cc::RenderPass;
3503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)using cc::RenderPassId;
3603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)using cc::DrawQuad;
3703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)using cc::TextureDrawQuad;
3803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)using cc::DelegatedFrameData;
3903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)using cc::CompositorFrame;
4003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
4103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)static void ContextLostThunk(void*) {
4203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  LOG(FATAL) << "Context lost";
4303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
4403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
4503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)ChildGLImpl::ChildGLImpl(ApplicationConnection* surfaces_service_connection,
4603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                         CommandBufferPtr command_buffer)
471320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    : start_time_(base::TimeTicks::Now()),
481320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      next_resource_id_(1),
491320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      weak_factory_(this) {
501320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  surfaces_service_connection->ConnectToService(&surfaces_service_);
511320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  surfaces_service_->CreateSurfaceConnection(base::Bind(
521320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci      &ChildGLImpl::SurfaceConnectionCreated, weak_factory_.GetWeakPtr()));
5303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  context_ =
5403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      MojoGLES2CreateContext(command_buffer.PassMessagePipe().release().value(),
5503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                             &ContextLostThunk,
5603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                             this,
5703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                             Environment::GetDefaultAsyncWaiter());
5803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  DCHECK(context_);
5903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  MojoGLES2MakeCurrent(context_);
6003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
6103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
6203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)ChildGLImpl::~ChildGLImpl() {
6303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  MojoGLES2DestroyContext(context_);
6403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  surface_->DestroySurface(mojo::SurfaceId::From(id_));
6503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
6603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
6703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)void ChildGLImpl::ProduceFrame(
6803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    ColorPtr color,
6903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    SizePtr size,
7003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    const mojo::Callback<void(SurfaceIdPtr id)>& callback) {
7103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  color_ = color.To<SkColor>();
7203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  size_ = size.To<gfx::Size>();
7303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  cube_.Init(size_.width(), size_.height());
7403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  cube_.set_color(
7503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      SkColorGetR(color_), SkColorGetG(color_), SkColorGetB(color_));
7603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  produce_callback_ = callback;
771320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  AllocateSurface();
7803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
7903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
801320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tuccivoid ChildGLImpl::SurfaceConnectionCreated(SurfacePtr surface,
811320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci                                           uint32_t id_namespace) {
821320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  surface_ = surface.Pass();
831320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  surface_.set_client(this);
8403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  allocator_.reset(new cc::SurfaceIdAllocator(id_namespace));
851320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci  AllocateSurface();
8603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
8703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
8803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)void ChildGLImpl::ReturnResources(Array<ReturnedResourcePtr> resources) {
8903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  for (size_t i = 0; i < resources.size(); ++i) {
9003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    cc::ReturnedResource res = resources[i].To<cc::ReturnedResource>();
9103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    GLuint returned_texture = id_to_tex_map_[res.id];
9203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    glDeleteTextures(1, &returned_texture);
9303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  }
9403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
9503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
9603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)void ChildGLImpl::AllocateSurface() {
9703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  if (produce_callback_.is_null() || !allocator_)
9803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)    return;
9903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
10003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  id_ = allocator_->GenerateId();
10103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  surface_->CreateSurface(mojo::SurfaceId::From(id_), mojo::Size::From(size_));
10203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  produce_callback_.Run(SurfaceId::From(id_));
10303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  Draw();
10403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
10503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
10603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)void ChildGLImpl::Draw() {
10703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  // First, generate a GL texture and draw the cube into it.
10803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  GLuint texture = 0u;
10903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glGenTextures(1, &texture);
11003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glBindTexture(GL_TEXTURE_2D, texture);
11103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glTexImage2D(GL_TEXTURE_2D,
11203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               0,
11303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               GL_RGBA,
11403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               size_.width(),
11503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               size_.height(),
11603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               0,
11703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               GL_RGBA,
11803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               GL_UNSIGNED_BYTE,
11903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)               0);
12003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
12103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  GLuint fbo = 0u;
12203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glGenFramebuffers(1, &fbo);
12303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glBindFramebuffer(GL_FRAMEBUFFER, fbo);
12403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glFramebufferTexture2D(
12503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, texture, 0);
12603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  DCHECK_EQ(static_cast<GLenum>(GL_FRAMEBUFFER_COMPLETE),
12703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)            glCheckFramebufferStatus(GL_FRAMEBUFFER));
12803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glClearColor(1, 0, 0, 0.5);
12903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  cube_.UpdateForTimeDelta(0.16f);
13003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  cube_.Draw();
13103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
13203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  // Then, put the texture into a mailbox.
13303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  gpu::Mailbox mailbox = gpu::Mailbox::Generate();
13403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  glProduceTextureCHROMIUM(GL_TEXTURE_2D, mailbox.name);
13503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  GLuint sync_point = glInsertSyncPointCHROMIUM();
13603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  gpu::MailboxHolder holder(mailbox, GL_TEXTURE_2D, sync_point);
13703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
13803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  // Then, put the mailbox into a TransferableResource
13903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  cc::TransferableResource resource;
14003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.id = next_resource_id_++;
14103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  id_to_tex_map_[resource.id] = texture;
14203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.format = cc::RGBA_8888;
14303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.filter = GL_LINEAR;
14403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.size = size_;
14503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.mailbox_holder = holder;
14603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.is_repeated = false;
14703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  resource.is_software = false;
14803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
14903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  gfx::Rect rect(size_);
15003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  RenderPassId id(1, 1);
15103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  scoped_ptr<RenderPass> pass = RenderPass::Create();
15203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  pass->SetNew(id, rect, rect, gfx::Transform());
15303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
15403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  CreateAndAppendSimpleSharedQuadState(pass.get(), gfx::Transform(), size_);
15503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
15603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  TextureDrawQuad* texture_quad =
15703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      pass->CreateAndAppendDrawQuad<TextureDrawQuad>();
15803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  float vertex_opacity[4] = {1.0f, 1.0f, 0.2f, 1.0f};
15903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  texture_quad->SetNew(pass->shared_quad_state_list.back(),
16003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       rect,
16103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       rect,
16203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       rect,
16303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       resource.id,
16403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       true,
16503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       gfx::PointF(),
16603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       gfx::PointF(1.f, 1.f),
16703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       SK_ColorBLUE,
16803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       vertex_opacity,
16903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)                       false);
17003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
17103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  scoped_ptr<DelegatedFrameData> delegated_frame_data(new DelegatedFrameData);
17203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  delegated_frame_data->render_pass_list.push_back(pass.Pass());
17303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  delegated_frame_data->resource_list.push_back(resource);
17403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
17503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  scoped_ptr<CompositorFrame> frame(new CompositorFrame);
17603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  frame->delegated_frame_data = delegated_frame_data.Pass();
17703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
17803b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  surface_->SubmitFrame(mojo::SurfaceId::From(id_), mojo::Frame::From(*frame));
17903b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
18003b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)  base::MessageLoop::current()->PostDelayedTask(
18103b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      FROM_HERE,
18203b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      base::Bind(&ChildGLImpl::Draw, base::Unretained(this)),
18303b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)      base::TimeDelta::FromMilliseconds(50));
18403b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}
18503b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)
18603b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}  // namespace examples
18703b57e008b61dfcb1fbad3aea950ae0e001748b0Torne (Richard Coles)}  // namespace mojo
188