1 /*
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 #include <chrono>
9 #include <err.h>
10 #include <iostream>
11 #include <memory>
12 #include <string>
13 #include <sys/types.h>
14 #include <sys/uio.h>
15 #include <sys/wait.h>
16 #include <thread>
17 #include <unistd.h>
18 
19 #include "SkGraphics.h"
20 #include "SkRemoteGlyphCache.h"
21 #include "SkScalerContext.h"
22 #include "SkSurface.h"
23 
24 static std::string gSkpName;
25 static bool gUseGpu = true;
26 static bool gPurgeFontCaches = true;
27 static bool gUseProcess = true;
28 
29 class ServerDiscardableManager : public SkStrikeServer::DiscardableHandleManager {
30 public:
31     ServerDiscardableManager() = default;
32     ~ServerDiscardableManager() override = default;
33 
createHandle()34     SkDiscardableHandleId createHandle() override { return ++nextHandleId; }
lockHandle(SkDiscardableHandleId handleId)35     bool lockHandle(SkDiscardableHandleId handleId) override {
36         return handleId > lastPurgedHandleId;
37     }
purgeAll()38     void purgeAll() { lastPurgedHandleId = nextHandleId; }
39 
40 private:
41     SkDiscardableHandleId nextHandleId = 0u;
42     SkDiscardableHandleId lastPurgedHandleId = 0u;
43 };
44 
45 class ClientDiscardableManager : public SkStrikeClient::DiscardableHandleManager {
46 public:
47     class ScopedPurgeCache {
48     public:
ScopedPurgeCache(ClientDiscardableManager * manager)49         ScopedPurgeCache(ClientDiscardableManager* manager) : fManager(manager) {
50             if (fManager) fManager->allowPurging = true;
51         }
~ScopedPurgeCache()52         ~ScopedPurgeCache() {
53             if (fManager) fManager->allowPurging = false;
54         }
55 
56     private:
57         ClientDiscardableManager* fManager;
58     };
59 
60     ClientDiscardableManager() = default;
61     ~ClientDiscardableManager() override = default;
62 
deleteHandle(SkDiscardableHandleId)63     bool deleteHandle(SkDiscardableHandleId) override { return allowPurging; }
64 
65 private:
66     bool allowPurging = false;
67 };
68 
write_SkData(int fd,const SkData & data)69 static bool write_SkData(int fd, const SkData& data) {
70     size_t size = data.size();
71     ssize_t bytesWritten = ::write(fd, &size, sizeof(size));
72     if (bytesWritten < 0) {
73         err(1,"Failed write %zu", size);
74         return false;
75     }
76 
77     bytesWritten = ::write(fd, data.data(), data.size());
78     if (bytesWritten < 0) {
79         err(1,"Failed write %zu", size);
80         return false;
81     }
82 
83     return true;
84 }
85 
read_SkData(int fd)86 static sk_sp<SkData> read_SkData(int fd) {
87     size_t size;
88     ssize_t readSize = ::read(fd, &size, sizeof(size));
89     if (readSize <= 0) {
90         if (readSize < 0) {
91             err(1, "Failed read %zu", size);
92         }
93         return nullptr;
94     }
95 
96     auto out = SkData::MakeUninitialized(size);
97     auto data = (uint8_t*)out->data();
98 
99     size_t totalRead = 0;
100     while (totalRead < size) {
101         ssize_t sizeRead;
102         sizeRead = ::read(fd, &data[totalRead], size - totalRead);
103         if (sizeRead <= 0) {
104             if (readSize < 0) {
105                 err(1, "Failed read %zu", size);
106             }
107             return nullptr;
108         }
109         totalRead += sizeRead;
110     }
111 
112     return out;
113 }
114 
115 class Timer {
116 public:
start()117     void start() {
118         fStart = std::chrono::high_resolution_clock::now();
119     }
120 
stop()121     void stop() {
122         auto end = std::chrono::high_resolution_clock::now();
123         fElapsedSeconds += end - fStart;
124     }
125 
elapsedSeconds()126     double elapsedSeconds() {
127         return fElapsedSeconds.count();
128     }
129 
130 private:
131     decltype(std::chrono::high_resolution_clock::now()) fStart;
132     std::chrono::duration<double>                       fElapsedSeconds{0.0};
133 };
134 
push_font_data(const SkPicture & pic,SkStrikeServer * strikeServer,int writeFd)135 static bool push_font_data(const SkPicture& pic, SkStrikeServer* strikeServer, int writeFd) {
136     SkMatrix deviceMatrix = SkMatrix::I();
137     const SkIRect bounds = pic.cullRect().round();
138     const SkSurfaceProps props(SkSurfaceProps::kLegacyFontHost_InitType);
139     SkTextBlobCacheDiffCanvas filter(bounds.width(), bounds.height(), deviceMatrix, props,
140                                      strikeServer);
141     pic.playback(&filter);
142 
143     std::vector<uint8_t> fontData;
144     strikeServer->writeStrikeData(&fontData);
145     auto data = SkData::MakeWithoutCopy(fontData.data(), fontData.size());
146     return write_SkData(writeFd, *data);
147 }
148 
final_draw(std::string outFilename,SkData * picData,SkStrikeClient * client,ClientDiscardableManager * discardableManager,int readFd,int writeFd)149 static void final_draw(std::string outFilename, SkData* picData, SkStrikeClient* client,
150                        ClientDiscardableManager* discardableManager, int readFd, int writeFd) {
151     SkDeserialProcs procs;
152     auto decode = [](const void* data, size_t length, void* ctx) -> sk_sp<SkTypeface> {
153         return reinterpret_cast<SkStrikeClient*>(ctx)->deserializeTypeface(data, length);
154     };
155     procs.fTypefaceProc = decode;
156     procs.fTypefaceCtx = client;
157 
158     auto pic = SkPicture::MakeFromData(picData, &procs);
159 
160     auto cullRect = pic->cullRect();
161     auto r = cullRect.round();
162 
163     auto s = SkSurface::MakeRasterN32Premul(r.width(), r.height());
164     auto c = s->getCanvas();
165     auto picUnderTest = SkPicture::MakeFromData(picData, &procs);
166 
167     Timer drawTime;
168     auto randomData = SkData::MakeUninitialized(1u);
169     for (int i = 0; i < 100; i++) {
170         if (gPurgeFontCaches) {
171             ClientDiscardableManager::ScopedPurgeCache purge(discardableManager);
172             SkGraphics::PurgeFontCache();
173             SkASSERT(SkGraphics::GetFontCacheUsed() == 0u);
174         }
175 
176         drawTime.start();
177         if (client != nullptr) {
178             // Kick the renderer to send us the fonts.
179             write_SkData(writeFd, *randomData);
180             auto fontData = read_SkData(readFd);
181             if (fontData && !fontData->isEmpty()) {
182                 if (!client->readStrikeData(fontData->data(), fontData->size()))
183                     SK_ABORT("Bad serialization");
184             }
185         }
186         c->drawPicture(picUnderTest);
187         drawTime.stop();
188     }
189 
190     std::cout << "useProcess: " << gUseProcess
191               << " useGPU: " << gUseGpu
192               << " purgeCache: " << gPurgeFontCaches << std::endl;
193     fprintf(stderr, "%s use GPU %s elapsed time %8.6f s\n", gSkpName.c_str(),
194             gUseGpu ? "true" : "false", drawTime.elapsedSeconds());
195 
196     auto i = s->makeImageSnapshot();
197     auto data = i->encodeToData();
198     SkFILEWStream f(outFilename.c_str());
199     f.write(data->data(), data->size());
200 }
201 
gpu(int readFd,int writeFd)202 static void gpu(int readFd, int writeFd) {
203 
204     if (gUseGpu) {
205         auto picData = read_SkData(readFd);
206         if (picData == nullptr) {
207             return;
208         }
209 
210         sk_sp<ClientDiscardableManager> discardableManager = sk_make_sp<ClientDiscardableManager>();
211         SkStrikeClient strikeClient(discardableManager);
212 
213         final_draw("test.png", picData.get(), &strikeClient, discardableManager.get(), readFd,
214                    writeFd);
215     }
216 
217     ::close(writeFd);
218     ::close(readFd);
219 
220     printf("GPU is exiting\n");
221 }
222 
renderer(const std::string & skpName,int readFd,int writeFd)223 static int renderer(
224     const std::string& skpName, int readFd, int writeFd)
225 {
226     ServerDiscardableManager discardableManager;
227     SkStrikeServer server(&discardableManager);
228     auto closeAll = [readFd, writeFd]() {
229         ::close(writeFd);
230         ::close(readFd);
231     };
232 
233     auto skpData = SkData::MakeFromFileName(skpName.c_str());
234     std::cout << "skp stream is " << skpData->size() << " bytes long " << std::endl;
235 
236     sk_sp<SkData> stream;
237     if (gUseGpu) {
238         auto pic = SkPicture::MakeFromData(skpData.get());
239         SkSerialProcs procs;
240         auto encode = [](SkTypeface* tf, void* ctx) -> sk_sp<SkData> {
241             return reinterpret_cast<SkStrikeServer*>(ctx)->serializeTypeface(tf);
242         };
243         procs.fTypefaceProc = encode;
244         procs.fTypefaceCtx = &server;
245 
246         stream = pic->serialize(&procs);
247 
248         if (!write_SkData(writeFd, *stream)) {
249             closeAll();
250             return 1;
251         }
252 
253         while (true) {
254             auto inBuffer = read_SkData(readFd);
255             if (inBuffer == nullptr) {
256                 closeAll();
257                 return 0;
258             }
259             if (gPurgeFontCaches) discardableManager.purgeAll();
260             push_font_data(*pic.get(), &server, writeFd);
261         }
262     } else {
263         stream = skpData;
264         final_draw("test-correct.png", stream.get(), nullptr, nullptr, -1, -1);
265         closeAll();
266         return 0;
267     }
268 }
269 
main(int argc,char ** argv)270 int main(int argc, char** argv) {
271     std::string skpName = argc > 1 ? std::string{argv[1]} : std::string{"skps/desk_nytimes.skp"};
272     int mode = argc > 2 ? atoi(argv[2]) : -1;
273     printf("skp: %s\n", skpName.c_str());
274 
275     gSkpName = skpName;
276 
277     enum direction : int {kRead = 0, kWrite = 1};
278 
279 
280     int render_to_gpu[2],
281         gpu_to_render[2];
282 
283     for (int m = 0; m < 8; m++) {
284         int r = pipe(render_to_gpu);
285         if (r < 0) {
286             perror("Can't write picture from render to GPU ");
287             return 1;
288         }
289         r = pipe(gpu_to_render);
290         if (r < 0) {
291             perror("Can't write picture from render to GPU ");
292             return 1;
293         }
294 
295         gPurgeFontCaches = (m & 4) == 4;
296         gUseGpu = (m & 2) == 2;
297         gUseProcess = (m & 1) == 1;
298 
299         if (mode >= 0 && mode < 8 && mode != m) {
300             continue;
301         }
302 
303         if (gUseProcess) {
304             pid_t child = fork();
305             SkGraphics::Init();
306 
307             if (child == 0) {
308                 close(gpu_to_render[kRead]);
309                 close(render_to_gpu[kWrite]);
310                 gpu(render_to_gpu[kRead], gpu_to_render[kWrite]);
311             } else {
312                 close(render_to_gpu[kRead]);
313                 close(gpu_to_render[kWrite]);
314                 renderer(skpName, gpu_to_render[kRead], render_to_gpu[kWrite]);
315                 waitpid(child, nullptr, 0);
316             }
317         } else {
318             SkGraphics::Init();
319             std::thread(gpu, render_to_gpu[kRead], gpu_to_render[kWrite]).detach();
320             renderer(skpName, gpu_to_render[kRead], render_to_gpu[kWrite]);
321         }
322     }
323 
324     return 0;
325 }
326 
327