aboutsummaryrefslogtreecommitdiffhomepage
path: root/tools/skiaserve
diff options
context:
space:
mode:
Diffstat (limited to 'tools/skiaserve')
-rw-r--r--tools/skiaserve/skiaserve.cpp146
1 files changed, 87 insertions, 59 deletions
diff --git a/tools/skiaserve/skiaserve.cpp b/tools/skiaserve/skiaserve.cpp
index aa6d803264..fa98d79287 100644
--- a/tools/skiaserve/skiaserve.cpp
+++ b/tools/skiaserve/skiaserve.cpp
@@ -9,6 +9,7 @@
#include "GrContextFactory.h"
#include "SkCanvas.h"
#include "SkCommandLineFlags.h"
+#include "SkJSONCanvas.h"
#include "SkPicture.h"
#include "SkStream.h"
#include "SkSurface.h"
@@ -32,8 +33,44 @@ DEFINE_bool(useTemplate, true, "whether or not to use the skdebugger template st
static const int kImageWidth = 1920;
static const int kImageHeight = 1080;
+// TODO move to template file
+SkString generateTemplate(SkString source) {
+ SkString debuggerTemplate;
+ debuggerTemplate.appendf(
+ "<!DOCTYPE html>\n"
+ "<html>\n"
+ "<head>\n"
+ " <title>SkDebugger</title>\n"
+ " <meta charset=\"utf-8\" />\n"
+ " <meta http-equiv=\"X-UA-Compatible\" content=\"IE=egde,chrome=1\">\n"
+ " <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n"
+ " <script src=\"%s/res/js/core.js\" type=\"text/javascript\" charset=\"utf-8\"></script>\n"
+ " <link href=\"%s/res/vul/elements.html\" rel=\"import\" />\n"
+ "</head>\n"
+ "<body class=\"fullbleed layout vertical\">\n"
+ " <debugger-app-sk>This is the app."
+ " </debugger-app-sk>\n"
+ "</body>\n"
+ "</html>", source.c_str(), source.c_str());
+ return debuggerTemplate;
+
+}
+
+struct UploadContext {
+ SkDynamicMemoryWStream fStream;
+ MHD_PostProcessor* fPostProcessor;
+ MHD_Connection* connection;
+};
+
+struct Request {
+ Request() : fUploadContext(nullptr) {}
+ UploadContext* fUploadContext;
+ SkAutoTUnref<SkData> fPNG;
+ SkAutoTUnref<SkPicture> fPicture;
+};
+
// TODO factor this out into functions, also handle CPU path
-SkData* setupAndDrawToCanvas(SkStream* stream, SkString* error) {
+bool setupAndDrawToCanvas(Request* request, SkString* error) {
GrContextOptions grContextOpts;
SkAutoTDelete<GrContextFactory> factory(new GrContextFactory(grContextOpts));
@@ -55,65 +92,32 @@ SkData* setupAndDrawToCanvas(SkStream* stream, SkString* error) {
gl->makeCurrent();
// draw
- SkAutoTUnref<SkPicture> pic(SkPicture::CreateFromStream(stream));
- if (pic.get() == nullptr) {
+ request->fPicture.reset(
+ SkPicture::CreateFromStream(request->fUploadContext->fStream.detachAsStream()));
+ if (!request->fPicture.get()) {
error->appendf("Could not create picture from stream.\n");
- return nullptr;
+ return false;
}
SkCanvas* canvas = surface->getCanvas();
- canvas->drawPicture(pic);
+ canvas->drawPicture(request->fPicture);
// capture pixels
SkBitmap bmp;
bmp.setInfo(canvas->imageInfo());
if (!canvas->readPixels(&bmp, 0, 0)) {
error->appendf("Can't read canvas pixels.\n");
- return nullptr;
+ return false;
}
// write to png
- SkData* data = SkImageEncoder::EncodeData(bmp, SkImageEncoder::kPNG_Type, 100);
- if (!data) {
+ request->fPNG.reset(SkImageEncoder::EncodeData(bmp, SkImageEncoder::kPNG_Type, 100));
+ if (!request->fPNG) {
error->appendf("Can't encode a PNG.\n");
- return nullptr;
+ return false;
}
- return data;
+ return true;
}
-// TODO move to template file
-SkString generateTemplate(SkString source) {
- SkString debuggerTemplate;
- debuggerTemplate.appendf(
- "<!DOCTYPE html>\n"
- "<html>\n"
- "<head>\n"
- " <title>SkDebugger</title>\n"
- " <meta charset=\"utf-8\" />\n"
- " <meta http-equiv=\"X-UA-Compatible\" content=\"IE=egde,chrome=1\">\n"
- " <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n"
- " <script src=\"%s/res/js/core.js\" type=\"text/javascript\" charset=\"utf-8\"></script>\n"
- " <link href=\"%s/res/vul/elements.html\" rel=\"import\" />\n"
- "</head>\n"
- "<body class=\"fullbleed layout vertical\">\n"
- " <debugger-app-sk>This is the app."
- " </debugger-app-sk>\n"
- "</body>\n"
- "</html>", source.c_str(), source.c_str());
- return debuggerTemplate;
-
-}
-
-struct UploadContext {
- SkDynamicMemoryWStream stream;
- MHD_PostProcessor* pp;
- MHD_Connection* connection;
-};
-
-struct Request {
- Request() : fUploadContext(nullptr) {}
- UploadContext* fUploadContext;
- SkAutoTUnref<SkData> fPNG;
-};
static const size_t kBufferSize = 1024;
@@ -124,23 +128,34 @@ static int process_upload_data(void* cls, enum MHD_ValueKind kind,
struct UploadContext* uc = reinterpret_cast<UploadContext*>(cls);
if (0 != size) {
- uc->stream.write(data, size);
+ uc->fStream.write(data, size);
}
return MHD_YES;
}
-static int SendImage(MHD_Connection* connection, const SkData* data) {
+static int SendData(MHD_Connection* connection, const SkData* data, const char* type) {
MHD_Response* response = MHD_create_response_from_buffer(data->size(),
const_cast<void*>(data->data()),
MHD_RESPMEM_MUST_COPY);
- MHD_add_response_header(response, "Content-Type", "image/png");
+ MHD_add_response_header(response, "Content-Type", type);
int ret = MHD_queue_response(connection, MHD_HTTP_OK, response);
MHD_destroy_response(response);
return ret;
}
-static int SendTemplate(MHD_Connection* connection) {
+static int SendJSON(MHD_Connection* connection, SkPicture* picture) {
+ SkDynamicMemoryWStream stream;
+ SkAutoTUnref<SkJSONCanvas> jsonCanvas(new SkJSONCanvas(kImageWidth, kImageHeight, stream));
+ jsonCanvas->drawPicture(picture);
+ jsonCanvas->finish();
+
+ SkAutoTUnref<SkData> data(stream.copyToData());
+ return SendData(connection, data, "application/json");
+}
+
+static int SendTemplate(MHD_Connection* connection, bool redirect = false,
+ const char* redirectUrl = nullptr) {
SkString debuggerTemplate = generateTemplate(SkString("http://debugger.skia.org"));
MHD_Response* response = MHD_create_response_from_buffer(
@@ -148,6 +163,11 @@ static int SendTemplate(MHD_Connection* connection) {
(void*) const_cast<char*>(debuggerTemplate.c_str()),
MHD_RESPMEM_MUST_COPY);
+ if (redirect) {
+ MHD_add_response_header (response, "Location", redirectUrl);
+ }
+
+
int ret = MHD_queue_response(connection, MHD_HTTP_OK, response);
MHD_destroy_response(response);
return ret;
@@ -170,8 +190,9 @@ int postHandler(Request* request, MHD_Connection* connection,
// TODO make this a method on request
uc = new UploadContext;
uc->connection = connection;
- uc->pp = MHD_create_post_processor(connection, kBufferSize, &process_upload_data, uc);
- SkASSERT(uc->pp);
+ uc->fPostProcessor = MHD_create_post_processor(connection, kBufferSize,
+ &process_upload_data, uc);
+ SkASSERT(uc->fPostProcessor);
request->fUploadContext = uc;
return MHD_YES;
@@ -179,33 +200,39 @@ int postHandler(Request* request, MHD_Connection* connection,
// in process upload
if (0 != *upload_data_size) {
- SkASSERT(uc->pp);
- MHD_post_process(uc->pp, upload_data, *upload_data_size);
+ SkASSERT(uc->fPostProcessor);
+ MHD_post_process(uc->fPostProcessor, upload_data, *upload_data_size);
*upload_data_size = 0;
return MHD_YES;
}
// end of upload
- MHD_destroy_post_processor(uc->pp);
- uc->pp = nullptr;
+ MHD_destroy_post_processor(uc->fPostProcessor);
+ uc->fPostProcessor = nullptr;
// TODO response
SkString error;
- SkData* data = setupAndDrawToCanvas(uc->stream.detachAsStream(), &error);
- if (!data) {
+ if (!setupAndDrawToCanvas(request, &error)) {
// TODO send error
return MHD_YES;
}
- request->fPNG.reset(data);
- return SendTemplate(connection);
+ return SendTemplate(connection, true, "/");
}
int imgHandler(Request* request, MHD_Connection* connection,
const char* upload_data, size_t* upload_data_size) {
if (request->fPNG.get()) {
SkData* data = request->fPNG.get();
- return SendImage(connection, data);
+ return SendData(connection, data, "image/png");
+ }
+ return MHD_NO;
+}
+
+int infoHandler(Request* request, MHD_Connection* connection,
+ const char* upload_data, size_t* upload_data_size) {
+ if (request->fPicture.get()) {
+ return SendJSON(connection, request->fPicture);
}
return MHD_NO;
}
@@ -217,6 +244,7 @@ public:
fHandlers.push_back({MHD_HTTP_METHOD_GET, "/", rootHandler});
fHandlers.push_back({MHD_HTTP_METHOD_POST, "/new", postHandler});
fHandlers.push_back({MHD_HTTP_METHOD_GET, "/img", imgHandler});
+ //fHandlers.push_back({MHD_HTTP_METHOD_GET, "/cmd", infoHandler});
}
// This is clearly not efficient for a large number of urls and handlers