blob: c6ce35a4737d940049ab4646da2a6dd6e20b045a [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*,
28 const GrProgramInfo&,
29 GrPrimitiveType);
Jim Van Verth1223e7f2019-02-28 17:38:35 -050030
Jim Van Verth75c53262019-04-26 12:23:51 -040031 // Finds or creates a compatible MTLDepthStencilState based on the GrStencilSettings.
32 GrMtlDepthStencil* findOrCreateCompatibleDepthStencilState(const GrStencilSettings&,
33 GrSurfaceOrigin);
34
35 // Finds or creates a compatible MTLSamplerState based on the GrSamplerState.
36 GrMtlSampler* findOrCreateCompatibleSampler(const GrSamplerState&, uint32_t maxMipLevel);
37
Jim Van Verthd44c0de2019-07-30 18:25:32 +000038 id<MTLBuffer> getDynamicBuffer(size_t size, size_t* offset);
39 void addBufferCompletionHandler(GrMtlCommandBuffer* cmdBuffer);
Jim Van Verth35a67eb2019-05-03 10:58:40 -040040
Jim Van Verthcf23f582019-05-22 09:46:57 -040041 // Destroy any cached resources. To be called before releasing the MtlDevice.
42 void destroyResources();
43
Timothy Liange30739a2018-07-31 10:51:17 -040044private:
Jim Van Verth1223e7f2019-02-28 17:38:35 -050045#ifdef SK_DEBUG
46#define GR_PIPELINE_STATE_CACHE_STATS
47#endif
48
49 class PipelineStateCache : public ::SkNoncopyable {
50 public:
51 PipelineStateCache(GrMtlGpu* gpu);
52 ~PipelineStateCache();
53
Jim Van Verthcf23f582019-05-22 09:46:57 -040054 void release();
Robert Phillips901aff02019-10-08 12:32:56 -040055 GrMtlPipelineState* refPipelineState(GrRenderTarget*, const GrProgramInfo&,
Jim Van Verth1223e7f2019-02-28 17:38:35 -050056 GrPrimitiveType);
57
58 private:
Jim Van Verth1223e7f2019-02-28 17:38:35 -050059 struct Entry;
60
61 struct DescHash {
62 uint32_t operator()(const GrProgramDesc& desc) const {
63 return SkOpts::hash_fn(desc.asKey(), desc.keyLength(), 0);
64 }
65 };
66
67 SkLRUCache<const GrMtlPipelineStateBuilder::Desc, std::unique_ptr<Entry>, DescHash> fMap;
68
69 GrMtlGpu* fGpu;
70
71#ifdef GR_PIPELINE_STATE_CACHE_STATS
72 int fTotalRequests;
73 int fCacheMisses;
74#endif
75 };
76
Jim Van Verthbbf85f92019-06-20 12:38:38 -040077 // Buffer allocator
78 class BufferSuballocator : public SkRefCnt {
79 public:
Jim Van Verthd44c0de2019-07-30 18:25:32 +000080 BufferSuballocator(id<MTLDevice> device, size_t size);
Jim Van Verthbbf85f92019-06-20 12:38:38 -040081 ~BufferSuballocator() {
82 fBuffer = nil;
83 fTotalSize = 0;
84 }
85
86 id<MTLBuffer> getAllocation(size_t size, size_t* offset);
87 void addCompletionHandler(GrMtlCommandBuffer* cmdBuffer);
88 size_t size() { return fTotalSize; }
89
90 private:
91 id<MTLBuffer> fBuffer;
92 size_t fTotalSize;
Jim Van Verth5e8f3892019-07-01 15:11:29 -040093 size_t fHead SK_GUARDED_BY(fMutex); // where we start allocating
94 size_t fTail SK_GUARDED_BY(fMutex); // where we start deallocating
Jim Van Verthbbf85f92019-06-20 12:38:38 -040095 SkSpinlock fMutex;
96 };
Jim Van Verthd44c0de2019-07-30 18:25:32 +000097 static constexpr size_t kBufferSuballocatorStartSize = 1024*1024;
Jim Van Verthbbf85f92019-06-20 12:38:38 -040098
Timothy Liange30739a2018-07-31 10:51:17 -040099 GrMtlGpu* fGpu;
Jim Van Verth1223e7f2019-02-28 17:38:35 -0500100
101 // Cache of GrMtlPipelineStates
102 std::unique_ptr<PipelineStateCache> fPipelineStateCache;
Jim Van Verth75c53262019-04-26 12:23:51 -0400103
104 SkTDynamicHash<GrMtlSampler, GrMtlSampler::Key> fSamplers;
105 SkTDynamicHash<GrMtlDepthStencil, GrMtlDepthStencil::Key> fDepthStencilStates;
Jim Van Verth35a67eb2019-05-03 10:58:40 -0400106
Jim Van Verthd44c0de2019-07-30 18:25:32 +0000107 // This is ref-counted because we might delete the GrContext before the command buffer
108 // finishes. The completion handler will retain a reference to this so it won't get
Jim Van Verthbbf85f92019-06-20 12:38:38 -0400109 // deleted along with the GrContext.
Jim Van Verthd44c0de2019-07-30 18:25:32 +0000110 sk_sp<BufferSuballocator> fBufferSuballocator;
Jim Van Verth22098812019-08-07 11:22:13 -0400111 size_t fBufferSuballocatorMaxSize;
Timothy Liange30739a2018-07-31 10:51:17 -0400112};
113
114#endif