1cb93a386Sopenharmony_ci/*
2cb93a386Sopenharmony_ci* Copyright 2018 Google Inc.
3cb93a386Sopenharmony_ci*
4cb93a386Sopenharmony_ci* Use of this source code is governed by a BSD-style license that can be
5cb93a386Sopenharmony_ci* found in the LICENSE file.
6cb93a386Sopenharmony_ci*/
7cb93a386Sopenharmony_ci
8cb93a386Sopenharmony_ci#include "src/gpu/mtl/GrMtlUniformHandler.h"
9cb93a386Sopenharmony_ci
10cb93a386Sopenharmony_ci#include "include/private/GrMtlTypesPriv.h"
11cb93a386Sopenharmony_ci#include "src/gpu/GrTexture.h"
12cb93a386Sopenharmony_ci#include "src/gpu/glsl/GrGLSLProgramBuilder.h"
13cb93a386Sopenharmony_ci
14cb93a386Sopenharmony_ci#if !__has_feature(objc_arc)
15cb93a386Sopenharmony_ci#error This file must be compiled with Arc. Use -fobjc-arc flag
16cb93a386Sopenharmony_ci#endif
17cb93a386Sopenharmony_ci
18cb93a386Sopenharmony_ciGR_NORETAIN_BEGIN
19cb93a386Sopenharmony_ci
20cb93a386Sopenharmony_ci// To determine whether a current offset is aligned, we can just 'and' the lowest bits with the
21cb93a386Sopenharmony_ci// alignment mask. A value of 0 means aligned, any other value is how many bytes past alignment we
22cb93a386Sopenharmony_ci// are. This works since all alignments are powers of 2. The mask is always (alignment - 1).
23cb93a386Sopenharmony_cistatic uint32_t grsltype_to_alignment_mask(GrSLType type) {
24cb93a386Sopenharmony_ci    switch(type) {
25cb93a386Sopenharmony_ci        case kInt_GrSLType:
26cb93a386Sopenharmony_ci        case kUInt_GrSLType:
27cb93a386Sopenharmony_ci        case kFloat_GrSLType:
28cb93a386Sopenharmony_ci            return 0x3;
29cb93a386Sopenharmony_ci        case kInt2_GrSLType:
30cb93a386Sopenharmony_ci        case kUInt2_GrSLType:
31cb93a386Sopenharmony_ci        case kFloat2_GrSLType:
32cb93a386Sopenharmony_ci            return 0x7;
33cb93a386Sopenharmony_ci        case kInt3_GrSLType:
34cb93a386Sopenharmony_ci        case kUInt3_GrSLType:
35cb93a386Sopenharmony_ci        case kFloat3_GrSLType:
36cb93a386Sopenharmony_ci        case kInt4_GrSLType:
37cb93a386Sopenharmony_ci        case kUInt4_GrSLType:
38cb93a386Sopenharmony_ci        case kFloat4_GrSLType:
39cb93a386Sopenharmony_ci            return 0xF;
40cb93a386Sopenharmony_ci
41cb93a386Sopenharmony_ci        case kFloat2x2_GrSLType:
42cb93a386Sopenharmony_ci            return 0x7;
43cb93a386Sopenharmony_ci        case kFloat3x3_GrSLType:
44cb93a386Sopenharmony_ci            return 0xF;
45cb93a386Sopenharmony_ci        case kFloat4x4_GrSLType:
46cb93a386Sopenharmony_ci            return 0xF;
47cb93a386Sopenharmony_ci
48cb93a386Sopenharmony_ci        case kShort_GrSLType:
49cb93a386Sopenharmony_ci        case kUShort_GrSLType:
50cb93a386Sopenharmony_ci        case kHalf_GrSLType:
51cb93a386Sopenharmony_ci            return 0x1;
52cb93a386Sopenharmony_ci        case kShort2_GrSLType:
53cb93a386Sopenharmony_ci        case kUShort2_GrSLType:
54cb93a386Sopenharmony_ci        case kHalf2_GrSLType:
55cb93a386Sopenharmony_ci            return 0x3;
56cb93a386Sopenharmony_ci        case kShort3_GrSLType:
57cb93a386Sopenharmony_ci        case kShort4_GrSLType:
58cb93a386Sopenharmony_ci        case kUShort3_GrSLType:
59cb93a386Sopenharmony_ci        case kUShort4_GrSLType:
60cb93a386Sopenharmony_ci        case kHalf3_GrSLType:
61cb93a386Sopenharmony_ci        case kHalf4_GrSLType:
62cb93a386Sopenharmony_ci            return 0x7;
63cb93a386Sopenharmony_ci
64cb93a386Sopenharmony_ci        case kHalf2x2_GrSLType:
65cb93a386Sopenharmony_ci            return 0x3;
66cb93a386Sopenharmony_ci        case kHalf3x3_GrSLType:
67cb93a386Sopenharmony_ci            return 0x7;
68cb93a386Sopenharmony_ci        case kHalf4x4_GrSLType:
69cb93a386Sopenharmony_ci            return 0x7;
70cb93a386Sopenharmony_ci
71cb93a386Sopenharmony_ci        // This query is only valid for certain types.
72cb93a386Sopenharmony_ci        case kVoid_GrSLType:
73cb93a386Sopenharmony_ci        case kBool_GrSLType:
74cb93a386Sopenharmony_ci        case kBool2_GrSLType:
75cb93a386Sopenharmony_ci        case kBool3_GrSLType:
76cb93a386Sopenharmony_ci        case kBool4_GrSLType:
77cb93a386Sopenharmony_ci        case kTexture2DSampler_GrSLType:
78cb93a386Sopenharmony_ci        case kTextureExternalSampler_GrSLType:
79cb93a386Sopenharmony_ci        case kTexture2DRectSampler_GrSLType:
80cb93a386Sopenharmony_ci        case kSampler_GrSLType:
81cb93a386Sopenharmony_ci        case kTexture2D_GrSLType:
82cb93a386Sopenharmony_ci        case kInput_GrSLType:
83cb93a386Sopenharmony_ci            break;
84cb93a386Sopenharmony_ci    }
85cb93a386Sopenharmony_ci    SK_ABORT("Unexpected type");
86cb93a386Sopenharmony_ci}
87cb93a386Sopenharmony_ci
88cb93a386Sopenharmony_ci/** Returns the size in bytes taken up in Metal buffers for GrSLTypes. */
89cb93a386Sopenharmony_cistatic inline uint32_t grsltype_to_mtl_size(GrSLType type) {
90cb93a386Sopenharmony_ci    switch(type) {
91cb93a386Sopenharmony_ci        case kInt_GrSLType:
92cb93a386Sopenharmony_ci        case kUInt_GrSLType:
93cb93a386Sopenharmony_ci        case kFloat_GrSLType:
94cb93a386Sopenharmony_ci            return 4;
95cb93a386Sopenharmony_ci        case kInt2_GrSLType:
96cb93a386Sopenharmony_ci        case kUInt2_GrSLType:
97cb93a386Sopenharmony_ci        case kFloat2_GrSLType:
98cb93a386Sopenharmony_ci            return 8;
99cb93a386Sopenharmony_ci        case kInt3_GrSLType:
100cb93a386Sopenharmony_ci        case kUInt3_GrSLType:
101cb93a386Sopenharmony_ci        case kFloat3_GrSLType:
102cb93a386Sopenharmony_ci        case kInt4_GrSLType:
103cb93a386Sopenharmony_ci        case kUInt4_GrSLType:
104cb93a386Sopenharmony_ci        case kFloat4_GrSLType:
105cb93a386Sopenharmony_ci            return 16;
106cb93a386Sopenharmony_ci
107cb93a386Sopenharmony_ci        case kFloat2x2_GrSLType:
108cb93a386Sopenharmony_ci            return 16;
109cb93a386Sopenharmony_ci        case kFloat3x3_GrSLType:
110cb93a386Sopenharmony_ci            return 48;
111cb93a386Sopenharmony_ci        case kFloat4x4_GrSLType:
112cb93a386Sopenharmony_ci            return 64;
113cb93a386Sopenharmony_ci
114cb93a386Sopenharmony_ci        case kShort_GrSLType:
115cb93a386Sopenharmony_ci        case kUShort_GrSLType:
116cb93a386Sopenharmony_ci        case kHalf_GrSLType:
117cb93a386Sopenharmony_ci            return 2;
118cb93a386Sopenharmony_ci        case kShort2_GrSLType:
119cb93a386Sopenharmony_ci        case kUShort2_GrSLType:
120cb93a386Sopenharmony_ci        case kHalf2_GrSLType:
121cb93a386Sopenharmony_ci            return 4;
122cb93a386Sopenharmony_ci        case kShort3_GrSLType:
123cb93a386Sopenharmony_ci        case kShort4_GrSLType:
124cb93a386Sopenharmony_ci        case kUShort3_GrSLType:
125cb93a386Sopenharmony_ci        case kUShort4_GrSLType:
126cb93a386Sopenharmony_ci        case kHalf3_GrSLType:
127cb93a386Sopenharmony_ci        case kHalf4_GrSLType:
128cb93a386Sopenharmony_ci            return 8;
129cb93a386Sopenharmony_ci
130cb93a386Sopenharmony_ci        case kHalf2x2_GrSLType:
131cb93a386Sopenharmony_ci            return 8;
132cb93a386Sopenharmony_ci        case kHalf3x3_GrSLType:
133cb93a386Sopenharmony_ci            return 24;
134cb93a386Sopenharmony_ci        case kHalf4x4_GrSLType:
135cb93a386Sopenharmony_ci            return 32;
136cb93a386Sopenharmony_ci
137cb93a386Sopenharmony_ci        // This query is only valid for certain types.
138cb93a386Sopenharmony_ci        case kVoid_GrSLType:
139cb93a386Sopenharmony_ci        case kBool_GrSLType:
140cb93a386Sopenharmony_ci        case kBool2_GrSLType:
141cb93a386Sopenharmony_ci        case kBool3_GrSLType:
142cb93a386Sopenharmony_ci        case kBool4_GrSLType:
143cb93a386Sopenharmony_ci        case kTexture2DSampler_GrSLType:
144cb93a386Sopenharmony_ci        case kTextureExternalSampler_GrSLType:
145cb93a386Sopenharmony_ci        case kTexture2DRectSampler_GrSLType:
146cb93a386Sopenharmony_ci        case kSampler_GrSLType:
147cb93a386Sopenharmony_ci        case kTexture2D_GrSLType:
148cb93a386Sopenharmony_ci        case kInput_GrSLType:
149cb93a386Sopenharmony_ci            break;
150cb93a386Sopenharmony_ci    }
151cb93a386Sopenharmony_ci    SK_ABORT("Unexpected type");
152cb93a386Sopenharmony_ci}
153cb93a386Sopenharmony_ci
154cb93a386Sopenharmony_ci// Given the current offset into the ubo, calculate the offset for the uniform we're trying to add
155cb93a386Sopenharmony_ci// taking into consideration all alignment requirements. The uniformOffset is set to the offset for
156cb93a386Sopenharmony_ci// the new uniform, and currentOffset is updated to be the offset to the end of the new uniform.
157cb93a386Sopenharmony_cistatic uint32_t get_ubo_aligned_offset(uint32_t* currentOffset,
158cb93a386Sopenharmony_ci                                       uint32_t* maxAlignment,
159cb93a386Sopenharmony_ci                                       GrSLType type,
160cb93a386Sopenharmony_ci                                       int arrayCount) {
161cb93a386Sopenharmony_ci    uint32_t alignmentMask = grsltype_to_alignment_mask(type);
162cb93a386Sopenharmony_ci    if (alignmentMask > *maxAlignment) {
163cb93a386Sopenharmony_ci        *maxAlignment = alignmentMask;
164cb93a386Sopenharmony_ci    }
165cb93a386Sopenharmony_ci    uint32_t offsetDiff = *currentOffset & alignmentMask;
166cb93a386Sopenharmony_ci    if (offsetDiff != 0) {
167cb93a386Sopenharmony_ci        offsetDiff = alignmentMask - offsetDiff + 1;
168cb93a386Sopenharmony_ci    }
169cb93a386Sopenharmony_ci    uint32_t uniformOffset = *currentOffset + offsetDiff;
170cb93a386Sopenharmony_ci    SkASSERT(sizeof(float) == 4);
171cb93a386Sopenharmony_ci    if (arrayCount) {
172cb93a386Sopenharmony_ci        *currentOffset = uniformOffset + grsltype_to_mtl_size(type) * arrayCount;
173cb93a386Sopenharmony_ci    } else {
174cb93a386Sopenharmony_ci        *currentOffset = uniformOffset + grsltype_to_mtl_size(type);
175cb93a386Sopenharmony_ci    }
176cb93a386Sopenharmony_ci    return uniformOffset;
177cb93a386Sopenharmony_ci}
178cb93a386Sopenharmony_ci
179cb93a386Sopenharmony_ciGrGLSLUniformHandler::UniformHandle GrMtlUniformHandler::internalAddUniformArray(
180cb93a386Sopenharmony_ci                                                                   const GrFragmentProcessor* owner,
181cb93a386Sopenharmony_ci                                                                   uint32_t visibility,
182cb93a386Sopenharmony_ci                                                                   GrSLType type,
183cb93a386Sopenharmony_ci                                                                   const char* name,
184cb93a386Sopenharmony_ci                                                                   bool mangleName,
185cb93a386Sopenharmony_ci                                                                   int arrayCount,
186cb93a386Sopenharmony_ci                                                                   const char** outName) {
187cb93a386Sopenharmony_ci    SkASSERT(name && strlen(name));
188cb93a386Sopenharmony_ci    SkASSERT(GrSLTypeCanBeUniformValue(type));
189cb93a386Sopenharmony_ci
190cb93a386Sopenharmony_ci    // TODO this is a bit hacky, lets think of a better way.  Basically we need to be able to use
191cb93a386Sopenharmony_ci    // the uniform view matrix name in the GP, and the GP is immutable so it has to tell the PB
192cb93a386Sopenharmony_ci    // exactly what name it wants to use for the uniform view matrix.  If we prefix anythings, then
193cb93a386Sopenharmony_ci    // the names will mismatch.  I think the correct solution is to have all GPs which need the
194cb93a386Sopenharmony_ci    // uniform view matrix, they should upload the view matrix in their setData along with regular
195cb93a386Sopenharmony_ci    // uniforms.
196cb93a386Sopenharmony_ci    char prefix = 'u';
197cb93a386Sopenharmony_ci    if ('u' == name[0] || !strncmp(name, GR_NO_MANGLE_PREFIX, strlen(GR_NO_MANGLE_PREFIX))) {
198cb93a386Sopenharmony_ci        prefix = '\0';
199cb93a386Sopenharmony_ci    }
200cb93a386Sopenharmony_ci    SkString resolvedName = fProgramBuilder->nameVariable(prefix, name, mangleName);
201cb93a386Sopenharmony_ci
202cb93a386Sopenharmony_ci    uint32_t offset = get_ubo_aligned_offset(&fCurrentUBOOffset, &fCurrentUBOMaxAlignment,
203cb93a386Sopenharmony_ci                                             type, arrayCount);
204cb93a386Sopenharmony_ci    SkString layoutQualifier;
205cb93a386Sopenharmony_ci    layoutQualifier.appendf("offset=%d", offset);
206cb93a386Sopenharmony_ci
207cb93a386Sopenharmony_ci    // When outputing the GLSL, only the outer uniform block will get the Uniform modifier. Thus
208cb93a386Sopenharmony_ci    // we set the modifier to none for all uniforms declared inside the block.
209cb93a386Sopenharmony_ci    MtlUniformInfo tempInfo;
210cb93a386Sopenharmony_ci    tempInfo.fVariable = GrShaderVar{std::move(resolvedName),
211cb93a386Sopenharmony_ci                                     type,
212cb93a386Sopenharmony_ci                                     GrShaderVar::TypeModifier::None,
213cb93a386Sopenharmony_ci                                     arrayCount,
214cb93a386Sopenharmony_ci                                     std::move(layoutQualifier),
215cb93a386Sopenharmony_ci                                     SkString()};
216cb93a386Sopenharmony_ci
217cb93a386Sopenharmony_ci    tempInfo.fVisibility = kFragment_GrShaderFlag | kVertex_GrShaderFlag;
218cb93a386Sopenharmony_ci    tempInfo.fOwner      = owner;
219cb93a386Sopenharmony_ci    tempInfo.fRawName    = SkString(name);
220cb93a386Sopenharmony_ci    tempInfo.fUBOffset   = offset;
221cb93a386Sopenharmony_ci
222cb93a386Sopenharmony_ci    fUniforms.push_back(tempInfo);
223cb93a386Sopenharmony_ci
224cb93a386Sopenharmony_ci    if (outName) {
225cb93a386Sopenharmony_ci        *outName = fUniforms.back().fVariable.c_str();
226cb93a386Sopenharmony_ci    }
227cb93a386Sopenharmony_ci
228cb93a386Sopenharmony_ci    return GrGLSLUniformHandler::UniformHandle(fUniforms.count() - 1);
229cb93a386Sopenharmony_ci}
230cb93a386Sopenharmony_ci
231cb93a386Sopenharmony_ciGrGLSLUniformHandler::SamplerHandle GrMtlUniformHandler::addSampler(
232cb93a386Sopenharmony_ci        const GrBackendFormat& backendFormat, GrSamplerState, const GrSwizzle& swizzle,
233cb93a386Sopenharmony_ci        const char* name, const GrShaderCaps* caps) {
234cb93a386Sopenharmony_ci    int binding = fSamplers.count();
235cb93a386Sopenharmony_ci
236cb93a386Sopenharmony_ci    SkASSERT(name && strlen(name));
237cb93a386Sopenharmony_ci
238cb93a386Sopenharmony_ci    constexpr char prefix = 'u';
239cb93a386Sopenharmony_ci    SkString mangleName = fProgramBuilder->nameVariable(prefix, name, /*mangle=*/true);
240cb93a386Sopenharmony_ci
241cb93a386Sopenharmony_ci    GrTextureType type = backendFormat.textureType();
242cb93a386Sopenharmony_ci
243cb93a386Sopenharmony_ci    SkString layoutQualifier;
244cb93a386Sopenharmony_ci    layoutQualifier.appendf("binding=%d", binding);
245cb93a386Sopenharmony_ci
246cb93a386Sopenharmony_ci    MtlUniformInfo tempInfo;
247cb93a386Sopenharmony_ci    tempInfo.fVariable = GrShaderVar{std::move(mangleName),
248cb93a386Sopenharmony_ci                                     GrSLCombinedSamplerTypeForTextureType(type),
249cb93a386Sopenharmony_ci                                     GrShaderVar::TypeModifier::Uniform,
250cb93a386Sopenharmony_ci                                     GrShaderVar::kNonArray,
251cb93a386Sopenharmony_ci                                     std::move(layoutQualifier),
252cb93a386Sopenharmony_ci                                     SkString()};
253cb93a386Sopenharmony_ci
254cb93a386Sopenharmony_ci    tempInfo.fVisibility = kFragment_GrShaderFlag;
255cb93a386Sopenharmony_ci    tempInfo.fOwner      = nullptr;
256cb93a386Sopenharmony_ci    tempInfo.fRawName    = SkString(name);
257cb93a386Sopenharmony_ci    tempInfo.fUBOffset   = 0;
258cb93a386Sopenharmony_ci
259cb93a386Sopenharmony_ci    fSamplers.push_back(tempInfo);
260cb93a386Sopenharmony_ci
261cb93a386Sopenharmony_ci    fSamplerSwizzles.push_back(swizzle);
262cb93a386Sopenharmony_ci    SkASSERT(fSamplerSwizzles.count() == fSamplers.count());
263cb93a386Sopenharmony_ci    return GrGLSLUniformHandler::SamplerHandle(fSamplers.count() - 1);
264cb93a386Sopenharmony_ci}
265cb93a386Sopenharmony_ci
266cb93a386Sopenharmony_civoid GrMtlUniformHandler::appendUniformDecls(GrShaderFlags visibility, SkString* out) const {
267cb93a386Sopenharmony_ci    for (const UniformInfo& sampler : fSamplers.items()) {
268cb93a386Sopenharmony_ci        SkASSERT(sampler.fVariable.getType() == kTexture2DSampler_GrSLType);
269cb93a386Sopenharmony_ci        if (visibility == sampler.fVisibility) {
270cb93a386Sopenharmony_ci            sampler.fVariable.appendDecl(fProgramBuilder->shaderCaps(), out);
271cb93a386Sopenharmony_ci            out->append(";\n");
272cb93a386Sopenharmony_ci        }
273cb93a386Sopenharmony_ci    }
274cb93a386Sopenharmony_ci
275cb93a386Sopenharmony_ci#ifdef SK_DEBUG
276cb93a386Sopenharmony_ci    bool firstOffsetCheck = false;
277cb93a386Sopenharmony_ci    for (const MtlUniformInfo& localUniform : fUniforms.items()) {
278cb93a386Sopenharmony_ci        if (!firstOffsetCheck) {
279cb93a386Sopenharmony_ci            // Check to make sure we are starting our offset at 0 so the offset qualifier we
280cb93a386Sopenharmony_ci            // set on each variable in the uniform block is valid.
281cb93a386Sopenharmony_ci            SkASSERT(0 == localUniform.fUBOffset);
282cb93a386Sopenharmony_ci            firstOffsetCheck = true;
283cb93a386Sopenharmony_ci        }
284cb93a386Sopenharmony_ci    }
285cb93a386Sopenharmony_ci#endif
286cb93a386Sopenharmony_ci
287cb93a386Sopenharmony_ci    SkString uniformsString;
288cb93a386Sopenharmony_ci    for (const UniformInfo& localUniform : fUniforms.items()) {
289cb93a386Sopenharmony_ci        if (visibility & localUniform.fVisibility) {
290cb93a386Sopenharmony_ci            if (GrSLTypeCanBeUniformValue(localUniform.fVariable.getType())) {
291cb93a386Sopenharmony_ci                localUniform.fVariable.appendDecl(fProgramBuilder->shaderCaps(), &uniformsString);
292cb93a386Sopenharmony_ci                uniformsString.append(";\n");
293cb93a386Sopenharmony_ci            }
294cb93a386Sopenharmony_ci        }
295cb93a386Sopenharmony_ci    }
296cb93a386Sopenharmony_ci
297cb93a386Sopenharmony_ci    if (!uniformsString.isEmpty()) {
298cb93a386Sopenharmony_ci        out->appendf("layout (binding=%d) uniform uniformBuffer\n{\n", kUniformBinding);
299cb93a386Sopenharmony_ci        out->appendf("%s\n};\n", uniformsString.c_str());
300cb93a386Sopenharmony_ci    }
301cb93a386Sopenharmony_ci}
302cb93a386Sopenharmony_ci
303cb93a386Sopenharmony_ciGR_NORETAIN_END
304