#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace mbgl { using namespace style; static gl::VertexVector tileVertices() { gl::VertexVector result; result.emplace_back(FillProgram::layoutVertex({ 0, 0 })); result.emplace_back(FillProgram::layoutVertex({ util::EXTENT, 0 })); result.emplace_back(FillProgram::layoutVertex({ 0, util::EXTENT })); result.emplace_back(FillProgram::layoutVertex({ util::EXTENT, util::EXTENT })); return result; } static gl::IndexVector tileTriangleIndices() { gl::IndexVector result; result.emplace_back(0, 1, 2); result.emplace_back(1, 2, 3); return result; } static gl::IndexVector tileLineStripIndices() { gl::IndexVector result; result.emplace_back(0); result.emplace_back(1); result.emplace_back(3); result.emplace_back(2); result.emplace_back(0); return result; } static gl::VertexVector rasterVertices() { gl::VertexVector result; result.emplace_back(RasterProgram::layoutVertex({ 0, 0 }, { 0, 0 })); result.emplace_back(RasterProgram::layoutVertex({ util::EXTENT, 0 }, { 32767, 0 })); result.emplace_back(RasterProgram::layoutVertex({ 0, util::EXTENT }, { 0, 32767 })); result.emplace_back(RasterProgram::layoutVertex({ util::EXTENT, util::EXTENT }, { 32767, 32767 })); return result; } Painter::Painter(gl::Context& context_, const TransformState& state_, float pixelRatio) : context(context_), state(state_), tileVertexBuffer(context.createVertexBuffer(tileVertices())), rasterVertexBuffer(context.createVertexBuffer(rasterVertices())), tileTriangleIndexBuffer(context.createIndexBuffer(tileTriangleIndices())), tileBorderIndexBuffer(context.createIndexBuffer(tileLineStripIndices())) { tileTriangleSegments.emplace_back(0, 0, 4, 6); tileBorderSegments.emplace_back(0, 0, 4, 5); rasterSegments.emplace_back(0, 0, 4, 6); gl::debugging::enable(); ProgramParameters programParameters{ pixelRatio, false }; programs = std::make_unique(context, programParameters); #ifndef NDEBUG ProgramParameters programParametersOverdraw{ pixelRatio, true }; overdrawPrograms = std::make_unique(context, programParametersOverdraw); #endif } Painter::~Painter() = default; bool Painter::needsAnimation() const { return frameHistory.needsAnimation(util::DEFAULT_FADE_DURATION); } void Painter::cleanup() { context.performCleanup(); } void Painter::render(const Style& style, const FrameData& frame_, View& view, SpriteAtlas& annotationSpriteAtlas) { frame = frame_; if (frame.contextMode == GLContextMode::Shared) { context.setDirtyState(); } PaintParameters parameters { #ifndef NDEBUG paintMode() == PaintMode::Overdraw ? *overdrawPrograms : *programs, #else *programs, #endif view }; glyphAtlas = style.glyphAtlas.get(); spriteAtlas = style.spriteAtlas.get(); lineAtlas = style.lineAtlas.get(); RenderData renderData = style.getRenderData(frame.debugOptions, state.getAngle()); const std::vector& order = renderData.order; const std::unordered_set& sources = renderData.sources; // Update the default matrices to the current viewport dimensions. state.getProjMatrix(projMatrix); pixelsToGLUnits = {{ 2.0f / state.getSize().width, -2.0f / state.getSize().height }}; if (state.getViewportMode() == ViewportMode::FlippedY) { pixelsToGLUnits[1] *= -1; } frameHistory.record(frame.timePoint, state.getZoom(), frame.mapMode == MapMode::Continuous ? util::DEFAULT_FADE_DURATION : Milliseconds(0)); // - UPLOAD PASS ------------------------------------------------------------------------------- // Uploads all required buffers and images before we do any actual rendering. { MBGL_DEBUG_GROUP("upload"); spriteAtlas->upload(context, 0); lineAtlas->upload(context, 0); glyphAtlas->upload(context, 0); frameHistory.upload(context, 0); annotationSpriteAtlas.upload(context, 0); for (const auto& item : order) { if (item.bucket && item.bucket->needsUpload()) { item.bucket->upload(context); } } } // - CLEAR ------------------------------------------------------------------------------------- // Renders the backdrop of the OpenGL view. This also paints in areas where we don't have any // tiles whatsoever. { MBGL_DEBUG_GROUP("clear"); view.bind(); context.clear(paintMode() == PaintMode::Overdraw ? Color::black() : renderData.backgroundColor, 1.0f, 0); } // - CLIPPING MASKS ---------------------------------------------------------------------------- // Draws the clipping masks to the stencil buffer. { MBGL_DEBUG_GROUP("clip"); // Update all clipping IDs. algorithm::ClipIDGenerator generator; for (const auto& source : sources) { source->baseImpl->startRender(generator, projMatrix, state); } MBGL_DEBUG_GROUP("clipping masks"); for (const auto& stencil : generator.getStencils()) { MBGL_DEBUG_GROUP(std::string{ "mask: " } + util::toString(stencil.first)); renderClippingMask(stencil.first, stencil.second); } } #if not MBGL_USE_GLES2 and not defined(NDEBUG) if (frame.debugOptions & MapDebugOptions::StencilClip) { renderClipMasks(parameters); return; } #endif // Actually render the layers if (debug::renderTree) { Log::Info(Event::Render, "{"); indent++; } // TODO: Correctly compute the number of layers recursively beforehand. depthRangeSize = 1 - (order.size() + 2) * numSublayers * depthEpsilon; // - OPAQUE PASS ------------------------------------------------------------------------------- // Render everything top-to-bottom by using reverse iterators. Render opaque objects first. renderPass(parameters, RenderPass::Opaque, order.rbegin(), order.rend(), 0, 1); // - TRANSLUCENT PASS -------------------------------------------------------------------------- // Make a second pass, rendering translucent objects. This time, we render bottom-to-top. renderPass(parameters, RenderPass::Translucent, order.begin(), order.end(), static_cast(order.size()) - 1, -1); if (debug::renderTree) { Log::Info(Event::Render, "}"); indent--; } // - DEBUG PASS -------------------------------------------------------------------------------- // Renders debug overlays. { MBGL_DEBUG_GROUP("debug"); // Finalize the rendering, e.g. by calling debug render calls per tile. // This guarantees that we have at least one function per tile called. // When only rendering layers via the stylesheet, it's possible that we don't // ever visit a tile during rendering. for (const auto& source : sources) { source->baseImpl->finishRender(*this); } } #if not MBGL_USE_GLES2 and not defined(NDEBUG) if (frame.debugOptions & MapDebugOptions::DepthBuffer) { renderDepthBuffer(parameters); } #endif // TODO: Find a better way to unbind VAOs after we're done with them without introducing // unnecessary bind(0)/bind(N) sequences. { MBGL_DEBUG_GROUP("cleanup"); context.activeTexture = 1; context.texture[1] = 0; context.activeTexture = 0; context.texture[0] = 0; context.vertexArrayObject = 0; } } template void Painter::renderPass(PaintParameters& parameters, RenderPass pass_, Iterator it, Iterator end, uint32_t i, int8_t increment) { pass = pass_; MBGL_DEBUG_GROUP(pass == RenderPass::Opaque ? "opaque" : "translucent"); if (debug::renderTree) { Log::Info(Event::Render, "%*s%s {", indent++ * 4, "", pass == RenderPass::Opaque ? "opaque" : "translucent"); } for (; it != end; ++it, i += increment) { currentLayer = i; const auto& item = *it; const Layer& layer = item.layer; if (!layer.baseImpl->hasRenderPass(pass)) continue; if (layer.is()) { MBGL_DEBUG_GROUP("background"); renderBackground(parameters, *layer.as()); } else if (layer.is()) { MBGL_DEBUG_GROUP(layer.baseImpl->id + " - custom"); // Reset GL state to a known state so the CustomLayer always has a clean slate. context.vertexArrayObject = 0; context.setDepthMode(depthModeForSublayer(0, gl::DepthMode::ReadOnly)); context.setStencilMode(gl::StencilMode::disabled()); context.setColorMode(colorModeForRenderPass()); layer.as()->impl->render(state); // Reset the view back to our original one, just in case the CustomLayer changed // the viewport or Framebuffer. parameters.view.bind(); context.setDirtyState(); } else { MBGL_DEBUG_GROUP(layer.baseImpl->id + " - " + util::toString(item.tile->id)); item.bucket->render(*this, parameters, layer, *item.tile); } } if (debug::renderTree) { Log::Info(Event::Render, "%*s%s", --indent * 4, "", "}"); } } mat4 Painter::matrixForTile(const UnwrappedTileID& tileID) { mat4 matrix; state.matrixFor(matrix, tileID); matrix::multiply(matrix, projMatrix, matrix); return matrix; } gl::DepthMode Painter::depthModeForSublayer(uint8_t n, gl::DepthMode::Mask mask) const { float nearDepth = ((1 + currentLayer) * numSublayers + n) * depthEpsilon; float farDepth = nearDepth + depthRangeSize; return gl::DepthMode { gl::DepthMode::LessEqual, mask, { nearDepth, farDepth } }; } gl::StencilMode Painter::stencilModeForClipping(const ClipID& id) const { return gl::StencilMode { gl::StencilMode::Equal { static_cast(id.mask.to_ulong()) }, static_cast(id.reference.to_ulong()), 0, gl::StencilMode::Keep, gl::StencilMode::Keep, gl::StencilMode::Replace }; } gl::ColorMode Painter::colorModeForRenderPass() const { if (paintMode() == PaintMode::Overdraw) { const float overdraw = 1.0f / 8.0f; return gl::ColorMode { gl::ColorMode::Add { gl::ColorMode::ConstantColor, gl::ColorMode::One }, Color { overdraw, overdraw, overdraw, 0.0f }, gl::ColorMode::Mask { true, true, true, true } }; } else if (pass == RenderPass::Translucent) { return gl::ColorMode::alphaBlended(); } else { return gl::ColorMode::unblended(); } } } // namespace mbgl