Skip to content

Commit 9ca0594

Browse files
authoredFeb 15, 2025··
[Depends on #8344] Implement ApiHandler functions and add JsonWriter (#8379)
* Introduce FrameGraphInfo class * Move the assignment into pimpl * Make ctors explicit * Add ctors to fg info structs * Revert the macro change to align with existing * Address the comments * Remove pimpl and move func def to .cc * Fix * Convert the FrameGraph to FrameGraphInfo * Initialize and update debug server on engine side * Fix compile error * Address the comments * Update * Use camelCase * Use camelCase * Update * Add JsonWriter to convert fginfo to json * Add getStatus api and implement it * Add increment * Implement GET apis * Fix compile error * Update * Use c_str_safe * Add operator== for FrameGraphInfo * Call updateFrameGraph in appropriate locations * Address the comments
1 parent 05906a5 commit 9ca0594

File tree

7 files changed

+300
-21
lines changed

7 files changed

+300
-21
lines changed
 

‎libs/fgviewer/CMakeLists.txt

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ set(SRCS
2323
src/ApiHandler.h
2424
src/DebugServer.cpp
2525
src/FrameGraphInfo.cpp
26+
src/JsonWriter.cpp
2627
)
2728

2829
# ==================================================================================================

‎libs/fgviewer/include/fgviewer/FrameGraphInfo.h

+15-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (C) 2024 The Android Open Source Project
2+
* Copyright (C) 2024 The Android Open Source Project
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -36,17 +36,25 @@ class FrameGraphInfo {
3636

3737
FrameGraphInfo(FrameGraphInfo const &) = delete;
3838

39+
bool operator==(const FrameGraphInfo& rhs) const;
40+
3941
struct Pass {
4042
Pass(utils::CString name, std::vector<ResourceId> reads,
4143
std::vector<ResourceId> writes);
4244

45+
bool operator==(const Pass& rhs) const;
46+
4347
utils::CString name;
4448
std::vector<ResourceId> reads;
4549
std::vector<ResourceId> writes;
4650
};
4751

4852
struct Resource {
53+
bool operator==(const Resource& rhs) const;
54+
4955
struct Property {
56+
bool operator==(const Property& rhs) const;
57+
5058
utils::CString name;
5159
utils::CString value;
5260
};
@@ -70,6 +78,12 @@ class FrameGraphInfo {
7078
// The incoming passes should be sorted by the execution order.
7179
void setPasses(std::vector<Pass> sortedPasses);
7280

81+
const char* getViewName() const;
82+
83+
const std::vector<Pass>& getPasses() const;
84+
85+
const std::unordered_map<ResourceId, Resource>& getResources() const;
86+
7387
private:
7488
utils::CString viewName;
7589
// The order of the passes in the vector indicates the execution

‎libs/fgviewer/src/ApiHandler.cpp

+84-16
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
#include "ApiHandler.h"
1919

2020
#include <fgviewer/DebugServer.h>
21+
#include <fgviewer/JsonWriter.h>
2122

2223
#include <utils/FixedCapacityVector.h>
2324
#include <utils/Log.h>
@@ -40,29 +41,96 @@ auto const error = [](int line, std::string const& uri) {
4041

4142
} // anonymous
4243

43-
bool ApiHandler::handleGetApiFgInfo(struct mg_connection* conn,
44-
struct mg_request_info const* request) {
45-
auto const softError = [conn, request](char const* msg) {
46-
utils::slog.e << "[fgviewer] DebugServer: " << msg << ": " << request->query_string << utils::io::endl;
47-
mg_printf(conn, kErrorHeader.data(), "application/txt");
48-
mg_write(conn, msg, strlen(msg));
44+
bool ApiHandler::handleGet(CivetServer* server, struct mg_connection* conn) {
45+
struct mg_request_info const* request = mg_get_request_info(conn);
46+
std::string const& uri = request->local_uri;
47+
48+
if (uri.find("/api/status") == 0) {
49+
return handleGetStatus(conn, request);
50+
}
51+
52+
if (uri == "/api/framegraphs") {
53+
std::unique_lock const lock(mServer->mViewsMutex);
54+
mg_printf(conn, kSuccessHeader.data(), "application/json");
55+
mg_printf(conn, "[");
56+
int index = 0;
57+
for (auto const& view: mServer->mViews) {
58+
bool const last = (++index) == mServer->mViews.size();
59+
60+
JsonWriter writer;
61+
if (!writer.writeFrameGraphInfo(view.second)) {
62+
return error(__LINE__, uri);
63+
}
64+
65+
mg_printf(conn, "{ \"fgid\": \"%8.8x\", %s } %s", view.first, writer.getJsonString(),
66+
last ? "" : ",");
67+
}
68+
mg_printf(conn, "]");
69+
return true;
70+
}
71+
72+
if (uri == "/api/framegraph") {
73+
const FrameGraphInfo* result = getFrameGraphInfo(request);
74+
if (!result) {
75+
return error(__LINE__, uri);
76+
}
77+
78+
JsonWriter writer;
79+
if (!writer.writeFrameGraphInfo(*result)) {
80+
return error(__LINE__, uri);
81+
}
82+
mg_printf(conn, kSuccessHeader.data(), "application/json");
83+
mg_printf(conn, "{ %s }", writer.getJsonString());
4984
return true;
50-
};
85+
}
5186

52-
// TODO: Implement the method
53-
return true;
87+
return error(__LINE__, uri);
5488
}
5589

56-
void ApiHandler::addFrameGraph(FrameGraphInfo const* framegraph) {
57-
// TODO: Implement the method
90+
void ApiHandler::updateFrameGraph(ViewHandle view_handle) {
91+
std::unique_lock const lock(mStatusMutex);
92+
snprintf(statusFrameGraphId, sizeof(statusFrameGraphId), "%8.8x", view_handle);
93+
mCurrentStatus++;
94+
mStatusCondition.notify_all();
5895
}
5996

97+
const FrameGraphInfo* ApiHandler::getFrameGraphInfo(struct mg_request_info const* request) {
98+
size_t const qlength = strlen(request->query_string);
99+
char fgid[9] = {};
100+
if (mg_get_var(request->query_string, qlength, "fgid", fgid, sizeof(fgid)) < 0) {
101+
return nullptr;
102+
}
103+
uint32_t const id = strtoul(fgid, nullptr, 16);
104+
std::unique_lock const lock(mServer->mViewsMutex);
105+
const auto it = mServer->mViews.find(id);
106+
return it == mServer->mViews.end()
107+
? nullptr
108+
: &(it->second);
109+
}
60110

61-
bool ApiHandler::handleGet(CivetServer* server, struct mg_connection* conn) {
62-
struct mg_request_info const* request = mg_get_request_info(conn);
63-
std::string const& uri = request->local_uri;
64-
65-
// TODO: Implement the method
111+
bool ApiHandler::handleGetStatus(struct mg_connection* conn,
112+
struct mg_request_info const* request) {
113+
char const* qstr = request->query_string;
114+
if (qstr && strcmp(qstr, "firstTime") == 0) {
115+
mg_printf(conn, kSuccessHeader.data(), "application/txt");
116+
mg_write(conn, "0", 1);
117+
return true;
118+
}
119+
120+
std::unique_lock<std::mutex> lock(mStatusMutex);
121+
uint64_t const currentStatusCount = mCurrentStatus;
122+
if (mStatusCondition.wait_for(lock, 10s,
123+
[this, currentStatusCount] {
124+
return currentStatusCount < mCurrentStatus;
125+
})) {
126+
mg_printf(conn, kSuccessHeader.data(), "application/txt");
127+
mg_write(conn, statusFrameGraphId, 8);
128+
} else {
129+
mg_printf(conn, kSuccessHeader.data(), "application/txt");
130+
// Use '1' to indicate a no-op. This ensures that we don't block forever if the client is
131+
// gone.
132+
mg_write(conn, "1", 1);
133+
}
66134
return true;
67135
}
68136

‎libs/fgviewer/src/ApiHandler.h

+19-3
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717
#ifndef FGVIEWER_APIHANDLER_H
1818
#define FGVIEWER_APIHANDLER_H
1919

20+
#include <fgviewer/DebugServer.h>
21+
2022
#include <CivetServer.h>
2123

2224
namespace filament::fgviewer {
@@ -27,7 +29,8 @@ struct FrameGraphInfo;
2729
// Handles the following REST requests, where {id} is an 8-digit hex string.
2830
//
2931
// GET /api/framegraphs
30-
// GET /api/framegraph?fg={viewname}
32+
// GET /api/framegraph?fg={fgid}
33+
// GET /api/status
3134
//
3235
class ApiHandler : public CivetHandler {
3336
public:
@@ -36,12 +39,25 @@ class ApiHandler : public CivetHandler {
3639
~ApiHandler() = default;
3740

3841
bool handleGet(CivetServer* server, struct mg_connection* conn);
39-
void addFrameGraph(FrameGraphInfo const* frameGraph);
42+
43+
void updateFrameGraph(ViewHandle view_handle);
4044

4145
private:
42-
bool handleGetApiFgInfo(struct mg_connection* conn, struct mg_request_info const* request);
46+
const FrameGraphInfo* getFrameGraphInfo(struct mg_request_info const* request);
47+
48+
bool handleGetStatus(struct mg_connection* conn,
49+
struct mg_request_info const* request);
4350

4451
DebugServer* mServer;
52+
53+
std::mutex mStatusMutex;
54+
std::condition_variable mStatusCondition;
55+
char statusFrameGraphId[9] = {};
56+
57+
// This variable is to implement a *hanging* effect for /api/status. The call to /api/status
58+
// will always block until statusMaterialId is updated again. The client is expected to keep
59+
// calling /api/status (a constant "pull" to simulate a push).
60+
uint64_t mCurrentStatus = 0;
4561
};
4662

4763
} // filament::fgviewer

‎libs/fgviewer/src/DebugServer.cpp

+12
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,7 @@ ViewHandle DebugServer::createView(utils::CString name) {
110110
std::unique_lock<utils::Mutex> lock(mViewsMutex);
111111
ViewHandle handle = mViewCounter++;
112112
mViews.emplace(handle, FrameGraphInfo(std::move(name)));
113+
mApiHandler->updateFrameGraph(handle);
113114

114115
return handle;
115116
}
@@ -121,8 +122,19 @@ void DebugServer::destroyView(ViewHandle h) {
121122

122123
void DebugServer::update(ViewHandle h, FrameGraphInfo info) {
123124
std::unique_lock<utils::Mutex> lock(mViewsMutex);
125+
const auto it = mViews.find(h);
126+
if (it == mViews.end()) {
127+
slog.w << "[fgviewer] Received update for unknown handle " << h;
128+
return;
129+
}
130+
131+
bool has_changed = !(it->second == info);
132+
if (!has_changed)
133+
return;
134+
124135
mViews.erase(h);
125136
mViews.emplace(h, std::move(info));
137+
mApiHandler->updateFrameGraph(h);
126138
}
127139

128140
} // namespace filament::fgviewer

‎libs/fgviewer/src/FrameGraphInfo.cpp

+32-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (C) 2024 The Android Open Source Project
2+
* Copyright (C) 2024 The Android Open Source Project
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -25,15 +25,33 @@ FrameGraphInfo::~FrameGraphInfo() = default;
2525

2626
FrameGraphInfo::FrameGraphInfo(FrameGraphInfo&& rhs) noexcept = default;
2727

28+
bool FrameGraphInfo::operator==(const FrameGraphInfo& rhs) const {
29+
return viewName == rhs.viewName
30+
&& passes == rhs.passes
31+
&& resources == rhs.resources;
32+
}
33+
2834
FrameGraphInfo::Pass::Pass(utils::CString name, std::vector<ResourceId> reads,
2935
std::vector<ResourceId> writes): name(std::move(name)),
3036
reads(std::move(reads)),
3137
writes(std::move(writes)) {}
3238

39+
bool FrameGraphInfo::Pass::operator==(const Pass& rhs) const {
40+
return name == rhs.name && reads == rhs.reads && writes == rhs.writes;
41+
}
42+
3343
FrameGraphInfo::Resource::Resource(ResourceId id, utils::CString name,
3444
std::vector<Property> properties): id(id),
3545
name(std::move(name)), properties(std::move(properties)) {}
3646

47+
bool FrameGraphInfo::Resource::operator==(const Resource& rhs) const {
48+
return id == rhs.id && name == rhs.name && properties == rhs.properties;
49+
}
50+
51+
bool FrameGraphInfo::Resource::Property::operator==(const Property &rhs) const {
52+
return name == rhs.name && value == rhs.value;
53+
}
54+
3755
void FrameGraphInfo::setResources(
3856
std::unordered_map<ResourceId, Resource> resources) {
3957
this->resources = std::move(resources);
@@ -43,4 +61,17 @@ void FrameGraphInfo::setPasses(std::vector<Pass> sortedPasses) {
4361
passes = std::move(sortedPasses);
4462
}
4563

64+
const char* FrameGraphInfo::getViewName() const {
65+
return viewName.c_str_safe();
66+
}
67+
68+
const std::vector<FrameGraphInfo::Pass>& FrameGraphInfo::getPasses() const {
69+
return passes;
70+
}
71+
72+
const std::unordered_map<ResourceId, FrameGraphInfo::Resource>&
73+
FrameGraphInfo::getResources() const {
74+
return resources;
75+
}
76+
4677
} // namespace filament::fgviewer

0 commit comments

Comments
 (0)