blob: c542145aa604f0b56dbbd64d6cb0d3f68d5678bf [file] [log] [blame]
Timothy Liange30739a2018-07-31 10:51:17 -04001/*
2 * Copyright 2018 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6*/
7
8#ifndef GrMtlResourceProvider_DEFINED
9#define GrMtlResourceProvider_DEFINED
10
Jim Van Verthbbf85f92019-06-20 12:38:38 -040011#include "include/private/SkSpinlock.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/private/SkTArray.h"
13#include "src/core/SkLRUCache.h"
Jim Van Verth75c53262019-04-26 12:23:51 -040014#include "src/gpu/mtl/GrMtlDepthStencil.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050015#include "src/gpu/mtl/GrMtlPipelineStateBuilder.h"
Jim Van Verth75c53262019-04-26 12:23:51 -040016#include "src/gpu/mtl/GrMtlSampler.h"
Timothy Liange30739a2018-07-31 10:51:17 -040017
Aaron O'Mullan829b6a02019-07-08 01:31:14 +020018#import <Metal/Metal.h>
Timothy Liange30739a2018-07-31 10:51:17 -040019
20class GrMtlGpu;
Jim Van Verthbbf85f92019-06-20 12:38:38 -040021class GrMtlCommandBuffer;
Timothy Liange30739a2018-07-31 10:51:17 -040022
23class GrMtlResourceProvider {
24public:
Jim Van Verth1223e7f2019-02-28 17:38:35 -050025 GrMtlResourceProvider(GrMtlGpu* gpu);
Timothy Liange30739a2018-07-31 10:51:17 -040026
Robert Phillips901aff02019-10-08 12:32:56 -040027 GrMtlPipelineState* findOrCreateCompatiblePipelineState(GrRenderTarget*,
Robert Phillipsfcaae482019-11-07 10:17:03 -050028 const GrProgramInfo&);
Jim Van Verth1223e7f2019-02-28 17:38:35 -050029
Jim Van Verth75c53262019-04-26 12:23:51 -040030 // Finds or creates a compatible MTLDepthStencilState based on the GrStencilSettings.
31 GrMtlDepthStencil* findOrCreateCompatibleDepthStencilState(const GrStencilSettings&,
32 GrSurfaceOrigin);
33
34 // Finds or creates a compatible MTLSamplerState based on the GrSamplerState.
Robert Phillipsf22c57d2019-10-14 14:16:02 -040035 GrMtlSampler* findOrCreateCompatibleSampler(const GrSamplerState&);
Jim Van Verth75c53262019-04-26 12:23:51 -040036
Jim Van Verthd44c0de2019-07-30 18:25:32 +000037 id<MTLBuffer> getDynamicBuffer(size_t size, size_t* offset);
38 void addBufferCompletionHandler(GrMtlCommandBuffer* cmdBuffer);
Jim Van Verth35a67eb2019-05-03 10:58:40 -040039
Jim Van Verthcf23f582019-05-22 09:46:57 -040040 // Destroy any cached resources. To be called before releasing the MtlDevice.
41 void destroyResources();
42
Timothy Liange30739a2018-07-31 10:51:17 -040043private:
Jim Van Verth1223e7f2019-02-28 17:38:35 -050044#ifdef SK_DEBUG
45#define GR_PIPELINE_STATE_CACHE_STATS
46#endif
47
48 class PipelineStateCache : public ::SkNoncopyable {
49 public:
50 PipelineStateCache(GrMtlGpu* gpu);
51 ~PipelineStateCache();
52
Jim Van Verthcf23f582019-05-22 09:46:57 -040053 void release();
Robert Phillipsfcaae482019-11-07 10:17:03 -050054 GrMtlPipelineState* refPipelineState(GrRenderTarget*, const GrProgramInfo&);
Jim Van Verth1223e7f2019-02-28 17:38:35 -050055
56 private:
Jim Van Verth1223e7f2019-02-28 17:38:35 -050057 struct Entry;
58
59 struct DescHash {
60 uint32_t operator()(const GrProgramDesc& desc) const {
61 return SkOpts::hash_fn(desc.asKey(), desc.keyLength(), 0);
62 }
63 };
64
65 SkLRUCache<const GrMtlPipelineStateBuilder::Desc, std::unique_ptr<Entry>, DescHash> fMap;
66
67 GrMtlGpu* fGpu;
68
69#ifdef GR_PIPELINE_STATE_CACHE_STATS
70 int fTotalRequests;
71 int fCacheMisses;
72#endif
73 };
74
Jim Van Verthbbf85f92019-06-20 12:38:38 -040075 // Buffer allocator
76 class BufferSuballocator : public SkRefCnt {
77 public:
Jim Van Verthd44c0de2019-07-30 18:25:32 +000078 BufferSuballocator(id<MTLDevice> device, size_t size);
Jim Van Verthbbf85f92019-06-20 12:38:38 -040079 ~BufferSuballocator() {
80 fBuffer = nil;
81 fTotalSize = 0;
82 }
83
84 id<MTLBuffer> getAllocation(size_t size, size_t* offset);
85 void addCompletionHandler(GrMtlCommandBuffer* cmdBuffer);
86 size_t size() { return fTotalSize; }
87
88 private:
89 id<MTLBuffer> fBuffer;
90 size_t fTotalSize;
Jim Van Verth5e8f3892019-07-01 15:11:29 -040091 size_t fHead SK_GUARDED_BY(fMutex); // where we start allocating
92 size_t fTail SK_GUARDED_BY(fMutex); // where we start deallocating
Jim Van Verthbbf85f92019-06-20 12:38:38 -040093 SkSpinlock fMutex;
94 };
Jim Van Verthd44c0de2019-07-30 18:25:32 +000095 static constexpr size_t kBufferSuballocatorStartSize = 1024*1024;
Jim Van Verthbbf85f92019-06-20 12:38:38 -040096
Timothy Liange30739a2018-07-31 10:51:17 -040097 GrMtlGpu* fGpu;
Jim Van Verth1223e7f2019-02-28 17:38:35 -050098
99 // Cache of GrMtlPipelineStates
100 std::unique_ptr<PipelineStateCache> fPipelineStateCache;
Jim Van Verth75c53262019-04-26 12:23:51 -0400101
102 SkTDynamicHash<GrMtlSampler, GrMtlSampler::Key> fSamplers;
103 SkTDynamicHash<GrMtlDepthStencil, GrMtlDepthStencil::Key> fDepthStencilStates;
Jim Van Verth35a67eb2019-05-03 10:58:40 -0400104
Jim Van Verthd44c0de2019-07-30 18:25:32 +0000105 // This is ref-counted because we might delete the GrContext before the command buffer
106 // finishes. The completion handler will retain a reference to this so it won't get
Jim Van Verthbbf85f92019-06-20 12:38:38 -0400107 // deleted along with the GrContext.
Jim Van Verthd44c0de2019-07-30 18:25:32 +0000108 sk_sp<BufferSuballocator> fBufferSuballocator;
Jim Van Verth22098812019-08-07 11:22:13 -0400109 size_t fBufferSuballocatorMaxSize;
Timothy Liange30739a2018-07-31 10:51:17 -0400110};
111
112#endif