#ifndef CC_TILES_TILE_MANAGER_H_
#define CC_TILES_TILE_MANAGER_H_
#include <stddef.h>
#include <stdint.h>
#include <memory>
#include <set>
#include <string>
#include <unordered_map>
#include <unordered_set>
#include <utility>
#include <vector>
#include "base/memory/raw_ptr.h"
#include "base/synchronization/waitable_event.h"
#include "base/task/sequenced_task_runner.h"
#include "cc/base/unique_notifier.h"
#include "cc/paint/target_color_params.h"
#include "cc/raster/raster_buffer_provider.h"
#include "cc/raster/raster_query_queue.h"
#include "cc/raster/raster_source.h"
#include "cc/resources/memory_history.h"
#include "cc/resources/resource_pool.h"
#include "cc/tiles/checker_image_tracker.h"
#include "cc/tiles/decoded_image_tracker.h"
#include "cc/tiles/eviction_tile_priority_queue.h"
#include "cc/tiles/image_controller.h"
#include "cc/tiles/raster_tile_priority_queue.h"
#include "cc/tiles/tile.h"
#include "cc/tiles/tile_draw_info.h"
#include "cc/tiles/tile_manager_settings.h"
#include "cc/tiles/tile_task_manager.h"
#include "ui/gfx/display_color_spaces.h"
#include "url/gurl.h"
namespace base {
namespace trace_event {
class ConvertableToTraceFormat;
class TracedValue;
}
}
namespace cc {
class ImageDecodeCache;
class TilesWithResourceIterator;
class CC_EXPORT TileManagerClient {
public:
virtual void NotifyReadyToActivate() = 0;
virtual void NotifyReadyToDraw() = 0;
virtual void NotifyAllTileTasksCompleted() = 0;
virtual void NotifyTileStateChanged(const Tile* tile) = 0;
virtual std::unique_ptr<RasterTilePriorityQueue> BuildRasterQueue(
TreePriority tree_priority,
RasterTilePriorityQueue::Type type) = 0;
virtual std::unique_ptr<EvictionTilePriorityQueue> BuildEvictionQueue(
TreePriority tree_priority) = 0;
virtual std::unique_ptr<TilesWithResourceIterator>
CreateTilesWithResourceIterator() = 0;
virtual void SetIsLikelyToRequireADraw(bool is_likely_to_require_a_draw) = 0;
virtual TargetColorParams GetTargetColorParams(
gfx::ContentColorUsage content_color_usage) const = 0;
virtual void RequestImplSideInvalidationForCheckerImagedTiles() = 0;
virtual size_t GetFrameIndexForImage(const PaintImage& paint_image,
WhichTree tree) const = 0;
virtual int GetMSAASampleCountForRaster(
const scoped_refptr<DisplayItemList>& display_list) = 0;
virtual bool HasPendingTree() = 0;
protected:
virtual ~TileManagerClient() {}
};
struct RasterTaskCompletionStats {
RasterTaskCompletionStats();
size_t completed_count;
size_t canceled_count;
};
std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
RasterTaskCompletionStatsAsValue(const RasterTaskCompletionStats& stats);
class CC_EXPORT TileManager : CheckerImageTrackerClient {
public:
TileManager(TileManagerClient* client,
base::SequencedTaskRunner* origin_task_runner,
scoped_refptr<base::SequencedTaskRunner> image_worker_task_runner,
size_t scheduled_raster_task_limit,
const TileManagerSettings& tile_manager_settings);
TileManager(const TileManager&) = delete;
~TileManager() override;
TileManager& operator=(const TileManager&) = delete;
bool PrepareTiles(const GlobalStateThatImpactsTilePriority& state);
void FinishTasksAndCleanUp();
void SetResources(ResourcePool* resource_pool,
ImageDecodeCache* image_decode_cache,
TaskGraphRunner* task_graph_runner,
RasterBufferProvider* raster_buffer_provider,
bool use_gpu_rasterization,
RasterQueryQueue* pending_raster_queries);
void PrepareToDraw();
void DidModifyTilePriorities();
std::unique_ptr<Tile> CreateTile(const Tile::CreateInfo& info,
int layer_id,
int source_frame_number,
int flags);
bool IsReadyToActivate() const;
bool IsReadyToDraw() const;
const PaintImageIdFlatSet& TakeImagesToInvalidateOnSyncTree();
void DidActivateSyncTree();
void ClearCheckerImageTracking(bool can_clear_decode_policy_tracking);
void SetCheckerImagingForceDisabled(bool force_disable);
std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
BasicStateAsValue() const;
void BasicStateAsValueInto(base::trace_event::TracedValue* dict) const;
const MemoryHistory::Entry& memory_stats_from_last_assign() const {
return memory_stats_from_last_assign_;
}
void InitializeTilesWithResourcesForTesting(const std::vector<Tile*>& tiles) {
for (size_t i = 0; i < tiles.size(); ++i) {
TileDrawInfo& draw_info = tiles[i]->draw_info();
ResourcePool::InUsePoolResource resource =
resource_pool_->AcquireResource(
tiles[i]->desired_texture_size(),
raster_buffer_provider_->GetFormat(),
client_->GetTargetColorParams(gfx::ContentColorUsage::kSRGB)
.color_space);
raster_buffer_provider_->AcquireBufferForRaster(
resource, 0, 0,
false,
false,
false);
if (resource.gpu_backing()) {
resource.gpu_backing()->mailbox =
gpu::Mailbox::GenerateForSharedImage();
resource.gpu_backing()->mailbox_sync_token.Set(
gpu::GPU_IO, gpu::CommandBufferId::FromUnsafeValue(1), 1);
}
bool exported = resource_pool_->PrepareForExport(resource);
DCHECK(exported);
draw_info.SetResource(std::move(resource), false, false);
draw_info.set_resource_ready_for_draw();
}
}
void ReleaseTileResourcesForTesting(const std::vector<Tile*>& tiles) {
for (size_t i = 0; i < tiles.size(); ++i) {
Tile* tile = tiles[i];
FreeResourcesForTile(tile);
}
}
void SetGlobalStateForTesting(
const GlobalStateThatImpactsTilePriority& state) {
global_state_ = state;
}
void SetTileTaskManagerForTesting(
std::unique_ptr<TileTaskManager> tile_task_manager) {
tile_task_manager_ = std::move(tile_task_manager);
}
void SetRasterBufferProviderForTesting(
RasterBufferProvider* raster_buffer_provider) {
raster_buffer_provider_ = raster_buffer_provider;
}
void SetPendingRasterQueriesForTesting(
RasterQueryQueue* pending_raster_queries) {
pending_raster_queries_ = pending_raster_queries;
}
std::vector<Tile*> AllTilesForTesting() const {
std::vector<Tile*> tiles;
for (auto& tile_pair : tiles_)
tiles.push_back(tile_pair.second);
return tiles;
}
void SetScheduledRasterTaskLimitForTesting(size_t limit) {
scheduled_raster_task_limit_ = limit;
}
void CheckIfMoreTilesNeedToBePreparedForTesting() {
CheckIfMoreTilesNeedToBePrepared();
}
void SetMoreTilesNeedToBeRasterizedForTesting() {
all_tiles_that_need_to_be_rasterized_are_scheduled_ = false;
}
void ResetSignalsForTesting();
bool HasScheduledTileTasksForTesting() const {
return has_scheduled_tile_tasks_;
}
void OnRasterTaskCompleted(Tile::Id tile_id,
ResourcePool::InUsePoolResource resource,
bool was_canceled);
void NeedsInvalidationForCheckerImagedTiles() override;
std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
ActivationStateAsValue();
void ActivationStateAsValueInto(base::trace_event::TracedValue* state) const;
int num_of_tiles_with_checker_images() const {
return num_of_tiles_with_checker_images_;
}
CheckerImageTracker& checker_image_tracker() {
return checker_image_tracker_;
}
DecodedImageTracker& decoded_image_tracker() {
return decoded_image_tracker_;
}
const std::vector<DrawImage>& decode_tasks_for_testing(Tile::Id id) {
return scheduled_draw_images_[id];
}
void set_active_url(const GURL& url) { active_url_ = url; }
std::string GetHungCommitDebugInfo() const;
protected:
friend class Tile;
void Release(Tile* tile);
Tile::Id GetUniqueTileId() { return ++next_tile_id_; }
private:
class MemoryUsage {
public:
MemoryUsage();
MemoryUsage(size_t memory_bytes, size_t resource_count);
static MemoryUsage FromConfig(const gfx::Size& size,
viz::SharedImageFormat format);
static MemoryUsage FromTile(const Tile* tile);
MemoryUsage& operator+=(const MemoryUsage& other);
MemoryUsage& operator-=(const MemoryUsage& other);
MemoryUsage operator-(const MemoryUsage& other);
bool Exceeds(const MemoryUsage& limit) const;
int64_t memory_bytes() const { return memory_bytes_; }
private:
int64_t memory_bytes_;
int resource_count_;
};
struct Signals {
bool activate_tile_tasks_completed = false;
bool draw_tile_tasks_completed = false;
bool all_tile_tasks_completed = false;
bool activate_gpu_work_completed = false;
bool draw_gpu_work_completed = false;
bool did_notify_ready_to_activate = false;
bool did_notify_ready_to_draw = false;
bool did_notify_all_tile_tasks_completed = false;
};
struct PrioritizedWorkToSchedule {
PrioritizedWorkToSchedule();
PrioritizedWorkToSchedule(PrioritizedWorkToSchedule&& other);
~PrioritizedWorkToSchedule();
std::vector<PrioritizedTile> tiles_to_raster;
std::vector<PrioritizedTile> tiles_to_process_for_images;
std::vector<DrawImage> extra_prepaint_images;
CheckerImageTracker::ImageDecodeQueue checker_image_decode_queue;
};
void FreeResourcesForOccludedTiles();
void FreeResourcesForTile(Tile* tile);
void FreeResourcesForTileAndNotifyClientIfTileWasReadyToDraw(Tile* tile);
scoped_refptr<TileTask> CreateRasterTask(
const PrioritizedTile& prioritized_tile,
const TargetColorParams& target_color_params,
PrioritizedWorkToSchedule* work_to_schedule);
std::unique_ptr<EvictionTilePriorityQueue>
FreeTileResourcesUntilUsageIsWithinLimit(
std::unique_ptr<EvictionTilePriorityQueue> eviction_priority_queue,
const MemoryUsage& limit,
MemoryUsage* usage);
std::unique_ptr<EvictionTilePriorityQueue>
FreeTileResourcesWithLowerPriorityUntilUsageIsWithinLimit(
std::unique_ptr<EvictionTilePriorityQueue> eviction_priority_queue,
const MemoryUsage& limit,
const TilePriority& oother_priority,
MemoryUsage* usage);
bool TilePriorityViolatesMemoryPolicy(const TilePriority& priority);
bool AreRequiredTilesReadyToDraw(RasterTilePriorityQueue::Type type) const;
void CheckIfMoreTilesNeedToBePrepared();
void MarkTilesOutOfMemory(
std::unique_ptr<RasterTilePriorityQueue> queue) const;
viz::SharedImageFormat DetermineFormat(const Tile* tile) const;
void DidFinishRunningTileTasksRequiredForActivation();
void DidFinishRunningTileTasksRequiredForDraw();
void DidFinishRunningAllTileTasks(bool has_pending_queries);
scoped_refptr<TileTask> CreateTaskSetFinishedTask(
void (TileManager::*callback)());
PrioritizedWorkToSchedule AssignGpuMemoryToTiles();
void ScheduleTasks(PrioritizedWorkToSchedule work_to_schedule);
void PartitionImagesForCheckering(
const PrioritizedTile& prioritized_tile,
const TargetColorParams& target_color_params,
std::vector<DrawImage>* sync_decoded_images,
std::vector<PaintImage>* checkered_images,
const gfx::Rect* invalidated_rect,
base::flat_map<PaintImage::Id, size_t>* image_to_frame_index = nullptr);
void AddCheckeredImagesToDecodeQueue(
const PrioritizedTile& prioritized_tile,
const TargetColorParams& target_color_params,
CheckerImageTracker::DecodeType decode_type,
CheckerImageTracker::ImageDecodeQueue* image_decode_queue);
std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
ScheduledTasksStateAsValue() const;
bool UsePartialRaster(int msaa_sample_count) const;
void FlushAndIssueSignals();
void CheckPendingGpuWorkAndIssueSignals();
void IssueSignals();
void ScheduleCheckRasterFinishedQueries();
void CheckRasterFinishedQueries();
bool ShouldRasterOccludedTiles() const;
raw_ptr<TileManagerClient, DanglingUntriaged> client_;
raw_ptr<base::SequencedTaskRunner> task_runner_;
raw_ptr<ResourcePool, DanglingUntriaged> resource_pool_;
std::unique_ptr<TileTaskManager> tile_task_manager_;
raw_ptr<RasterBufferProvider, DanglingUntriaged> raster_buffer_provider_;
GlobalStateThatImpactsTilePriority global_state_;
size_t scheduled_raster_task_limit_;
const TileManagerSettings tile_manager_settings_;
bool use_gpu_rasterization_;
raw_ptr<RasterQueryQueue> pending_raster_queries_ = nullptr;
std::unordered_map<Tile::Id, Tile*> tiles_;
bool all_tiles_that_need_to_be_rasterized_are_scheduled_;
MemoryHistory::Entry memory_stats_from_last_assign_;
bool did_check_for_completed_tasks_since_last_schedule_tasks_;
bool did_oom_on_last_assign_;
ImageController image_controller_;
DecodedImageTracker decoded_image_tracker_;
CheckerImageTracker checker_image_tracker_;
RasterTaskCompletionStats raster_task_completion_stats_;
TaskGraph graph_;
UniqueNotifier more_tiles_need_prepare_check_notifier_;
Signals signals_;
UniqueNotifier signals_check_notifier_;
bool has_scheduled_tile_tasks_;
uint64_t prepare_tiles_count_;
uint64_t next_tile_id_;
std::unordered_set<Tile*> pending_gpu_work_tiles_;
uint64_t pending_required_for_activation_callback_id_ = 0;
uint64_t pending_required_for_draw_callback_id_ = 0;
bool pending_tile_requirements_dirty_ = false;
std::unordered_map<Tile::Id, std::vector<DrawImage>> scheduled_draw_images_;
std::vector<scoped_refptr<TileTask>> locked_image_tasks_;
int num_of_tiles_with_checker_images_ = 0;
GURL active_url_;
bool has_pending_queries_ = false;
base::CancelableOnceClosure check_pending_tile_queries_callback_;
base::WaitableEvent shutdown_event_;
base::WeakPtrFactory<TileManager> task_set_finished_weak_ptr_factory_{this};
base::WeakPtrFactory<TileManager> ready_to_draw_callback_weak_ptr_factory_{
this};
};
}
#endif