| /* | 
 |  * Copyright 2018 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 |  | 
 | #include <chrono> | 
 | #include <err.h> | 
 | #include <iostream> | 
 | #include <memory> | 
 | #include <string> | 
 | #include <sys/types.h> | 
 | #include <sys/uio.h> | 
 | #include <sys/wait.h> | 
 | #include <thread> | 
 | #include <unistd.h> | 
 |  | 
 | #include "SkGraphics.h" | 
 | #include "SkRemoteGlyphCache.h" | 
 | #include "SkScalerContext.h" | 
 | #include "SkSurface.h" | 
 |  | 
 | static std::string gSkpName; | 
 | static bool gUseGpu = true; | 
 | static bool gPurgeFontCaches = true; | 
 | static bool gUseProcess = true; | 
 |  | 
 | class ServerDiscardableManager : public SkStrikeServer::DiscardableHandleManager { | 
 | public: | 
 |     ServerDiscardableManager() = default; | 
 |     ~ServerDiscardableManager() override = default; | 
 |  | 
 |     SkDiscardableHandleId createHandle() override { return ++nextHandleId; } | 
 |     bool lockHandle(SkDiscardableHandleId handleId) override { | 
 |         return handleId > lastPurgedHandleId; | 
 |     } | 
 |     void purgeAll() { lastPurgedHandleId = nextHandleId; } | 
 |  | 
 | private: | 
 |     SkDiscardableHandleId nextHandleId = 0u; | 
 |     SkDiscardableHandleId lastPurgedHandleId = 0u; | 
 | }; | 
 |  | 
 | class ClientDiscardableManager : public SkStrikeClient::DiscardableHandleManager { | 
 | public: | 
 |     class ScopedPurgeCache { | 
 |     public: | 
 |         ScopedPurgeCache(ClientDiscardableManager* manager) : fManager(manager) { | 
 |             if (fManager) fManager->allowPurging = true; | 
 |         } | 
 |         ~ScopedPurgeCache() { | 
 |             if (fManager) fManager->allowPurging = false; | 
 |         } | 
 |  | 
 |     private: | 
 |         ClientDiscardableManager* fManager; | 
 |     }; | 
 |  | 
 |     ClientDiscardableManager() = default; | 
 |     ~ClientDiscardableManager() override = default; | 
 |  | 
 |     bool deleteHandle(SkDiscardableHandleId) override { return allowPurging; } | 
 |  | 
 | private: | 
 |     bool allowPurging = false; | 
 | }; | 
 |  | 
 | static bool write_SkData(int fd, const SkData& data) { | 
 |     size_t size = data.size(); | 
 |     ssize_t bytesWritten = ::write(fd, &size, sizeof(size)); | 
 |     if (bytesWritten < 0) { | 
 |         err(1,"Failed write %zu", size); | 
 |         return false; | 
 |     } | 
 |  | 
 |     bytesWritten = ::write(fd, data.data(), data.size()); | 
 |     if (bytesWritten < 0) { | 
 |         err(1,"Failed write %zu", size); | 
 |         return false; | 
 |     } | 
 |  | 
 |     return true; | 
 | } | 
 |  | 
 | static sk_sp<SkData> read_SkData(int fd) { | 
 |     size_t size; | 
 |     ssize_t readSize = ::read(fd, &size, sizeof(size)); | 
 |     if (readSize <= 0) { | 
 |         if (readSize < 0) { | 
 |             err(1, "Failed read %zu", size); | 
 |         } | 
 |         return nullptr; | 
 |     } | 
 |  | 
 |     auto out = SkData::MakeUninitialized(size); | 
 |     auto data = (uint8_t*)out->data(); | 
 |  | 
 |     size_t totalRead = 0; | 
 |     while (totalRead < size) { | 
 |         ssize_t sizeRead; | 
 |         sizeRead = ::read(fd, &data[totalRead], size - totalRead); | 
 |         if (sizeRead <= 0) { | 
 |             if (readSize < 0) { | 
 |                 err(1, "Failed read %zu", size); | 
 |             } | 
 |             return nullptr; | 
 |         } | 
 |         totalRead += sizeRead; | 
 |     } | 
 |  | 
 |     return out; | 
 | } | 
 |  | 
 | class Timer { | 
 | public: | 
 |     void start() { | 
 |         fStart = std::chrono::high_resolution_clock::now(); | 
 |     } | 
 |  | 
 |     void stop() { | 
 |         auto end = std::chrono::high_resolution_clock::now(); | 
 |         fElapsedSeconds += end - fStart; | 
 |     } | 
 |  | 
 |     double elapsedSeconds() { | 
 |         return fElapsedSeconds.count(); | 
 |     } | 
 |  | 
 | private: | 
 |     decltype(std::chrono::high_resolution_clock::now()) fStart; | 
 |     std::chrono::duration<double>                       fElapsedSeconds{0.0}; | 
 | }; | 
 |  | 
 | static bool push_font_data(const SkPicture& pic, SkStrikeServer* strikeServer, int writeFd) { | 
 |     SkMatrix deviceMatrix = SkMatrix::I(); | 
 |     const SkIRect bounds = pic.cullRect().round(); | 
 |     const SkSurfaceProps props(SkSurfaceProps::kLegacyFontHost_InitType); | 
 |     SkTextBlobCacheDiffCanvas filter(bounds.width(), bounds.height(), deviceMatrix, props, | 
 |                                      strikeServer); | 
 |     pic.playback(&filter); | 
 |  | 
 |     std::vector<uint8_t> fontData; | 
 |     strikeServer->writeStrikeData(&fontData); | 
 |     auto data = SkData::MakeWithoutCopy(fontData.data(), fontData.size()); | 
 |     return write_SkData(writeFd, *data); | 
 | } | 
 |  | 
 | static void final_draw(std::string outFilename, SkData* picData, SkStrikeClient* client, | 
 |                        ClientDiscardableManager* discardableManager, int readFd, int writeFd) { | 
 |     SkDeserialProcs procs; | 
 |     auto decode = [](const void* data, size_t length, void* ctx) -> sk_sp<SkTypeface> { | 
 |         return reinterpret_cast<SkStrikeClient*>(ctx)->deserializeTypeface(data, length); | 
 |     }; | 
 |     procs.fTypefaceProc = decode; | 
 |     procs.fTypefaceCtx = client; | 
 |  | 
 |     auto pic = SkPicture::MakeFromData(picData, &procs); | 
 |  | 
 |     auto cullRect = pic->cullRect(); | 
 |     auto r = cullRect.round(); | 
 |  | 
 |     auto s = SkSurface::MakeRasterN32Premul(r.width(), r.height()); | 
 |     auto c = s->getCanvas(); | 
 |     auto picUnderTest = SkPicture::MakeFromData(picData, &procs); | 
 |  | 
 |     Timer drawTime; | 
 |     auto randomData = SkData::MakeUninitialized(1u); | 
 |     for (int i = 0; i < 100; i++) { | 
 |         if (gPurgeFontCaches) { | 
 |             ClientDiscardableManager::ScopedPurgeCache purge(discardableManager); | 
 |             SkGraphics::PurgeFontCache(); | 
 |             SkASSERT(SkGraphics::GetFontCacheUsed() == 0u); | 
 |         } | 
 |  | 
 |         drawTime.start(); | 
 |         if (client != nullptr) { | 
 |             // Kick the renderer to send us the fonts. | 
 |             write_SkData(writeFd, *randomData); | 
 |             auto fontData = read_SkData(readFd); | 
 |             if (fontData && !fontData->isEmpty()) { | 
 |                 if (!client->readStrikeData(fontData->data(), fontData->size())) | 
 |                     SK_ABORT("Bad serialization"); | 
 |             } | 
 |         } | 
 |         c->drawPicture(picUnderTest); | 
 |         drawTime.stop(); | 
 |     } | 
 |  | 
 |     std::cout << "useProcess: " << gUseProcess | 
 |               << " useGPU: " << gUseGpu | 
 |               << " purgeCache: " << gPurgeFontCaches << std::endl; | 
 |     fprintf(stderr, "%s use GPU %s elapsed time %8.6f s\n", gSkpName.c_str(), | 
 |             gUseGpu ? "true" : "false", drawTime.elapsedSeconds()); | 
 |  | 
 |     auto i = s->makeImageSnapshot(); | 
 |     auto data = i->encodeToData(); | 
 |     SkFILEWStream f(outFilename.c_str()); | 
 |     f.write(data->data(), data->size()); | 
 | } | 
 |  | 
 | static void gpu(int readFd, int writeFd) { | 
 |  | 
 |     if (gUseGpu) { | 
 |         auto picData = read_SkData(readFd); | 
 |         if (picData == nullptr) { | 
 |             return; | 
 |         } | 
 |  | 
 |         sk_sp<ClientDiscardableManager> discardableManager = sk_make_sp<ClientDiscardableManager>(); | 
 |         SkStrikeClient strikeClient(discardableManager); | 
 |  | 
 |         final_draw("test.png", picData.get(), &strikeClient, discardableManager.get(), readFd, | 
 |                    writeFd); | 
 |     } | 
 |  | 
 |     ::close(writeFd); | 
 |     ::close(readFd); | 
 |  | 
 |     printf("GPU is exiting\n"); | 
 | } | 
 |  | 
 | static int renderer( | 
 |     const std::string& skpName, int readFd, int writeFd) | 
 | { | 
 |     ServerDiscardableManager discardableManager; | 
 |     SkStrikeServer server(&discardableManager); | 
 |     auto closeAll = [readFd, writeFd]() { | 
 |         ::close(writeFd); | 
 |         ::close(readFd); | 
 |     }; | 
 |  | 
 |     auto skpData = SkData::MakeFromFileName(skpName.c_str()); | 
 |     std::cout << "skp stream is " << skpData->size() << " bytes long " << std::endl; | 
 |  | 
 |     sk_sp<SkData> stream; | 
 |     if (gUseGpu) { | 
 |         auto pic = SkPicture::MakeFromData(skpData.get()); | 
 |         SkSerialProcs procs; | 
 |         auto encode = [](SkTypeface* tf, void* ctx) -> sk_sp<SkData> { | 
 |             return reinterpret_cast<SkStrikeServer*>(ctx)->serializeTypeface(tf); | 
 |         }; | 
 |         procs.fTypefaceProc = encode; | 
 |         procs.fTypefaceCtx = &server; | 
 |  | 
 |         stream = pic->serialize(&procs); | 
 |  | 
 |         if (!write_SkData(writeFd, *stream)) { | 
 |             closeAll(); | 
 |             return 1; | 
 |         } | 
 |  | 
 |         while (true) { | 
 |             auto inBuffer = read_SkData(readFd); | 
 |             if (inBuffer == nullptr) { | 
 |                 closeAll(); | 
 |                 return 0; | 
 |             } | 
 |             if (gPurgeFontCaches) discardableManager.purgeAll(); | 
 |             push_font_data(*pic.get(), &server, writeFd); | 
 |         } | 
 |     } else { | 
 |         stream = skpData; | 
 |         final_draw("test-correct.png", stream.get(), nullptr, nullptr, -1, -1); | 
 |         closeAll(); | 
 |         return 0; | 
 |     } | 
 | } | 
 |  | 
 | int main(int argc, char** argv) { | 
 |     std::string skpName = argc > 1 ? std::string{argv[1]} : std::string{"skps/desk_nytimes.skp"}; | 
 |     int mode = argc > 2 ? atoi(argv[2]) : -1; | 
 |     printf("skp: %s\n", skpName.c_str()); | 
 |  | 
 |     gSkpName = skpName; | 
 |  | 
 |     enum direction : int {kRead = 0, kWrite = 1}; | 
 |  | 
 |  | 
 |     int render_to_gpu[2], | 
 |         gpu_to_render[2]; | 
 |  | 
 |     for (int m = 0; m < 8; m++) { | 
 |         int r = pipe(render_to_gpu); | 
 |         if (r < 0) { | 
 |             perror("Can't write picture from render to GPU "); | 
 |             return 1; | 
 |         } | 
 |         r = pipe(gpu_to_render); | 
 |         if (r < 0) { | 
 |             perror("Can't write picture from render to GPU "); | 
 |             return 1; | 
 |         } | 
 |  | 
 |         gPurgeFontCaches = (m & 4) == 4; | 
 |         gUseGpu = (m & 2) == 2; | 
 |         gUseProcess = (m & 1) == 1; | 
 |  | 
 |         if (mode >= 0 && mode < 8 && mode != m) { | 
 |             continue; | 
 |         } | 
 |  | 
 |         if (gUseProcess) { | 
 |             pid_t child = fork(); | 
 |             SkGraphics::Init(); | 
 |  | 
 |             if (child == 0) { | 
 |                 close(gpu_to_render[kRead]); | 
 |                 close(render_to_gpu[kWrite]); | 
 |                 gpu(render_to_gpu[kRead], gpu_to_render[kWrite]); | 
 |             } else { | 
 |                 close(render_to_gpu[kRead]); | 
 |                 close(gpu_to_render[kWrite]); | 
 |                 renderer(skpName, gpu_to_render[kRead], render_to_gpu[kWrite]); | 
 |                 waitpid(child, nullptr, 0); | 
 |             } | 
 |         } else { | 
 |             SkGraphics::Init(); | 
 |             std::thread(gpu, render_to_gpu[kRead], gpu_to_render[kWrite]).detach(); | 
 |             renderer(skpName, gpu_to_render[kRead], render_to_gpu[kWrite]); | 
 |         } | 
 |     } | 
 |  | 
 |     return 0; | 
 | } | 
 |  |