blob: 764d1efcc8f405eff542a135b10002282357f8fa [file] [log] [blame]
Nader Jawad5f0a8002023-02-21 17:00:51 -08001/*
2 * Copyright (C) 2022 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef MESH_H_
18#define MESH_H_
19
20#include <GrDirectContext.h>
21#include <SkMesh.h>
Kevin Lubick9cc7f772023-07-20 17:11:30 +000022#include <include/gpu/ganesh/SkMeshGanesh.h>
Nader Jawad5f0a8002023-02-21 17:00:51 -080023#include <jni.h>
24#include <log/log.h>
25
26#include <utility>
27
28class MeshUniformBuilder {
29public:
30 struct MeshUniform {
31 template <typename T>
32 std::enable_if_t<std::is_trivially_copyable<T>::value, MeshUniform> operator=(
33 const T& val) {
34 if (!fVar) {
35 LOG_FATAL("Assigning to missing variable");
36 } else if (sizeof(val) != fVar->sizeInBytes()) {
37 LOG_FATAL("Incorrect value size");
38 } else {
39 void* dst = reinterpret_cast<void*>(
40 reinterpret_cast<uint8_t*>(fOwner->writableUniformData()) + fVar->offset);
41 memcpy(dst, &val, sizeof(val));
42 }
43 }
44
45 MeshUniform& operator=(const SkMatrix& val) {
46 if (!fVar) {
47 LOG_FATAL("Assigning to missing variable");
48 } else if (fVar->sizeInBytes() != 9 * sizeof(float)) {
49 LOG_FATAL("Incorrect value size");
50 } else {
51 float* data = reinterpret_cast<float*>(
52 reinterpret_cast<uint8_t*>(fOwner->writableUniformData()) + fVar->offset);
53 data[0] = val.get(0);
54 data[1] = val.get(3);
55 data[2] = val.get(6);
56 data[3] = val.get(1);
57 data[4] = val.get(4);
58 data[5] = val.get(7);
59 data[6] = val.get(2);
60 data[7] = val.get(5);
61 data[8] = val.get(8);
62 }
63 return *this;
64 }
65
66 template <typename T>
67 bool set(const T val[], const int count) {
68 static_assert(std::is_trivially_copyable<T>::value, "Value must be trivial copyable");
69 if (!fVar) {
70 LOG_FATAL("Assigning to missing variable");
71 return false;
72 } else if (sizeof(T) * count != fVar->sizeInBytes()) {
73 LOG_FATAL("Incorrect value size");
74 return false;
75 } else {
76 void* dst = reinterpret_cast<void*>(
77 reinterpret_cast<uint8_t*>(fOwner->writableUniformData()) + fVar->offset);
78 memcpy(dst, val, sizeof(T) * count);
79 }
80 return true;
81 }
82
83 MeshUniformBuilder* fOwner;
84 const SkRuntimeEffect::Uniform* fVar;
85 };
86 MeshUniform uniform(std::string_view name) { return {this, fMeshSpec->findUniform(name)}; }
87
88 explicit MeshUniformBuilder(sk_sp<SkMeshSpecification> meshSpec) {
89 fMeshSpec = sk_sp(meshSpec);
90 fUniforms = (SkData::MakeZeroInitialized(meshSpec->uniformSize()));
91 }
92
93 sk_sp<SkData> fUniforms;
94
95private:
96 void* writableUniformData() {
97 if (!fUniforms->unique()) {
98 fUniforms = SkData::MakeWithCopy(fUniforms->data(), fUniforms->size());
99 }
100 return fUniforms->writable_data();
101 }
102
103 sk_sp<SkMeshSpecification> fMeshSpec;
104};
105
106class Mesh {
107public:
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800108 Mesh(const sk_sp<SkMeshSpecification>& meshSpec, int mode,
109 std::vector<uint8_t>&& vertexBufferData, jint vertexCount, jint vertexOffset,
Nader Jawad5f0a8002023-02-21 17:00:51 -0800110 std::unique_ptr<MeshUniformBuilder> builder, const SkRect& bounds)
111 : mMeshSpec(meshSpec)
112 , mMode(mode)
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800113 , mVertexBufferData(std::move(vertexBufferData))
Nader Jawad5f0a8002023-02-21 17:00:51 -0800114 , mVertexCount(vertexCount)
115 , mVertexOffset(vertexOffset)
116 , mBuilder(std::move(builder))
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800117 , mBounds(bounds) {}
Nader Jawad5f0a8002023-02-21 17:00:51 -0800118
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800119 Mesh(const sk_sp<SkMeshSpecification>& meshSpec, int mode,
120 std::vector<uint8_t>&& vertexBufferData, jint vertexCount, jint vertexOffset,
121 std::vector<uint8_t>&& indexBuffer, jint indexCount, jint indexOffset,
Nader Jawad5f0a8002023-02-21 17:00:51 -0800122 std::unique_ptr<MeshUniformBuilder> builder, const SkRect& bounds)
123 : mMeshSpec(meshSpec)
124 , mMode(mode)
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800125 , mVertexBufferData(std::move(vertexBufferData))
Nader Jawad5f0a8002023-02-21 17:00:51 -0800126 , mVertexCount(vertexCount)
127 , mVertexOffset(vertexOffset)
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800128 , mIndexBufferData(std::move(indexBuffer))
Nader Jawad5f0a8002023-02-21 17:00:51 -0800129 , mIndexCount(indexCount)
130 , mIndexOffset(indexOffset)
131 , mBuilder(std::move(builder))
Nader Jawad8c1d7aa2023-03-02 15:59:11 -0800132 , mBounds(bounds) {}
Nader Jawad5f0a8002023-02-21 17:00:51 -0800133
134 Mesh(Mesh&&) = default;
135
136 Mesh& operator=(Mesh&&) = default;
137
138 [[nodiscard]] std::tuple<bool, SkString> validate();
139
140 void updateSkMesh(GrDirectContext* context) const {
141 GrDirectContext::DirectContextID genId = GrDirectContext::DirectContextID();
142 if (context) {
143 genId = context->directContextID();
144 }
145
146 if (mIsDirty || genId != mGenerationId) {
Kevin Lubick9cc7f772023-07-20 17:11:30 +0000147 auto vertexData = reinterpret_cast<const void*>(mVertexBufferData.data());
148#ifdef __ANDROID__
149 auto vb = SkMeshes::MakeVertexBuffer(context,
150 vertexData,
151 mVertexBufferData.size());
152#else
153 auto vb = SkMeshes::MakeVertexBuffer(vertexData,
154 mVertexBufferData.size());
155#endif
Nader Jawad5f0a8002023-02-21 17:00:51 -0800156 auto meshMode = SkMesh::Mode(mMode);
157 if (!mIndexBufferData.empty()) {
Kevin Lubick9cc7f772023-07-20 17:11:30 +0000158 auto indexData = reinterpret_cast<const void*>(mIndexBufferData.data());
159#ifdef __ANDROID__
160 auto ib = SkMeshes::MakeIndexBuffer(context,
161 indexData,
162 mIndexBufferData.size());
163#else
164 auto ib = SkMeshes::MakeIndexBuffer(indexData,
165 mIndexBufferData.size());
166#endif
Nader Jawad5f0a8002023-02-21 17:00:51 -0800167 mMesh = SkMesh::MakeIndexed(mMeshSpec, meshMode, vb, mVertexCount, mVertexOffset,
168 ib, mIndexCount, mIndexOffset, mBuilder->fUniforms,
169 mBounds)
170 .mesh;
171 } else {
172 mMesh = SkMesh::Make(mMeshSpec, meshMode, vb, mVertexCount, mVertexOffset,
173 mBuilder->fUniforms, mBounds)
174 .mesh;
175 }
176 mIsDirty = false;
177 mGenerationId = genId;
178 }
179 }
180
181 SkMesh& getSkMesh() const {
182 LOG_FATAL_IF(mIsDirty,
183 "Attempt to obtain SkMesh when Mesh is dirty, did you "
184 "forget to call updateSkMesh with a GrDirectContext? "
185 "Defensively creating a CPU mesh");
186 return mMesh;
187 }
188
189 void markDirty() { mIsDirty = true; }
190
191 MeshUniformBuilder* uniformBuilder() { return mBuilder.get(); }
192
193private:
Nader Jawad5f0a8002023-02-21 17:00:51 -0800194 sk_sp<SkMeshSpecification> mMeshSpec;
195 int mMode = 0;
196
197 std::vector<uint8_t> mVertexBufferData;
198 size_t mVertexCount = 0;
199 size_t mVertexOffset = 0;
200
201 std::vector<uint8_t> mIndexBufferData;
202 size_t mIndexCount = 0;
203 size_t mIndexOffset = 0;
204
205 std::unique_ptr<MeshUniformBuilder> mBuilder;
206 SkRect mBounds{};
207
208 mutable SkMesh mMesh{};
209 mutable bool mIsDirty = true;
210 mutable GrDirectContext::DirectContextID mGenerationId = GrDirectContext::DirectContextID();
211};
212#endif // MESH_H_