mirror of
https://github.com/overte-org/overte.git
synced 2025-04-06 22:32:57 +02:00
exposing feedback of timing
This commit is contained in:
parent
ab04a34ccf
commit
d6b6645630
14 changed files with 147 additions and 69 deletions
|
@ -4287,6 +4287,8 @@ void Application::idle() {
|
|||
}
|
||||
|
||||
{
|
||||
workload::Timings timings(1, PerformanceTimer::getTimerRecord("/idle/update/simulation").getAverage());
|
||||
_gameWorkload.updateSimulationTimings(timings);
|
||||
_gameWorkload.updateViews(_viewFrustum, getMyAvatar()->getHeadPosition());
|
||||
_gameWorkload._engine->run();
|
||||
}
|
||||
|
|
|
@ -489,9 +489,9 @@ void Stats::updateStats(bool force) {
|
|||
};
|
||||
for (int32_t j = 0; j < categories.size(); ++j) {
|
||||
QString recordKey = "/idle/update/" + categories[j];
|
||||
itr = allRecords.find(recordKey);
|
||||
if (itr != allRecords.end()) {
|
||||
float dt = (float)itr.value().getMovingAverage() / (float)USECS_PER_MSEC;
|
||||
auto record = PerformanceTimer::getTimerRecord(recordKey);
|
||||
if (record.getCount()) {
|
||||
float dt = (float) record.getMovingAverage() / (float)USECS_PER_MSEC;
|
||||
QString message = QString("\n %1 = %2").arg(categories[j]).arg(dt);
|
||||
idleUpdateStats.push(SortableStat(message, dt));
|
||||
}
|
||||
|
|
|
@ -15,6 +15,41 @@
|
|||
|
||||
#include "PhysicsBoundary.h"
|
||||
|
||||
void ControlViews::run(const workload::WorkloadContextPointer& renderContext, const Input& inputs, Output& outputs) {
|
||||
const auto& inViews = inputs.getN<0>();
|
||||
const auto& inTimings = inputs[1];
|
||||
|
||||
outputs = inViews;
|
||||
}
|
||||
|
||||
class WorkloadEngineBuilder {
|
||||
public:
|
||||
public:
|
||||
using Inputs = workload::VaryingSet2<workload::Views, workload::Timings>;
|
||||
using Outputs = workload::RegionTracker::Outputs;
|
||||
using JobModel = workload::Task::ModelIO<WorkloadEngineBuilder, Inputs, Outputs>;
|
||||
void build(JobModel& model, const workload::Varying& in, workload::Varying& out) {
|
||||
|
||||
const auto& inViews = in.getN<Inputs>(0);
|
||||
const auto& inTimings = in.getN<Inputs>(1);
|
||||
|
||||
const auto usedViews = model.addJob<workload::SetupViews>("setupViews", inViews);
|
||||
|
||||
ControlViews::Input controlViewsIn;
|
||||
controlViewsIn[0] = usedViews;
|
||||
controlViewsIn[1] = inTimings;
|
||||
const auto fixedViews = model.addJob<ControlViews>("controlViews", controlViewsIn);
|
||||
|
||||
const auto regionTrackerOut = model.addJob<workload::SpaceClassifierTask>("spaceClassifier", fixedViews);
|
||||
|
||||
model.addJob<PhysicsBoundary>("PhysicsBoundary", regionTrackerOut);
|
||||
|
||||
model.addJob<GameSpaceToRender>("SpaceToRender");
|
||||
|
||||
out = regionTrackerOut;
|
||||
}
|
||||
};
|
||||
|
||||
GameWorkloadContext::GameWorkloadContext(const workload::SpacePointer& space,
|
||||
const render::ScenePointer& scene,
|
||||
const PhysicalEntitySimulationPointer& simulation): WorkloadContext(space),
|
||||
|
@ -27,7 +62,7 @@ GameWorkloadContext::~GameWorkloadContext() {
|
|||
|
||||
|
||||
GameWorkload::GameWorkload() :
|
||||
_engine(std::make_shared<workload::Engine>(std::make_shared<workload::Task>(workload::SpaceClassifierTask::JobModel::create("Engine"))))
|
||||
_engine(std::make_shared<workload::Engine>(WorkloadEngineBuilder::JobModel::create("Engine")))
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -39,12 +74,6 @@ void GameWorkload::startup(const workload::SpacePointer& space,
|
|||
const render::ScenePointer& scene,
|
||||
const PhysicalEntitySimulationPointer& simulation) {
|
||||
_engine->reset(std::make_shared<GameWorkloadContext>(space, scene, simulation));
|
||||
|
||||
auto output = _engine->_task->getOutput();
|
||||
_engine->_task->addJob<GameSpaceToRender>("SpaceToRender");
|
||||
|
||||
const auto regionChanges = _engine->_task->getOutput();
|
||||
_engine->_task->addJob<PhysicsBoundary>("PhysicsBoundary", regionChanges);
|
||||
}
|
||||
|
||||
void GameWorkload::shutdown() {
|
||||
|
@ -55,9 +84,9 @@ void GameWorkload::updateViews(const ViewFrustum& frustum, const glm::vec3& head
|
|||
workload::Views views;
|
||||
views.emplace_back(workload::View::evalFromFrustum(frustum, headPosition - frustum.getPosition()));
|
||||
views.emplace_back(workload::View::evalFromFrustum(frustum));
|
||||
_engine->_task->feedInput(views);
|
||||
_engine->feedInput<WorkloadEngineBuilder::Inputs>(0, views);
|
||||
}
|
||||
|
||||
void GameWorkload::updateSimulationTimings(const workload::Timings& timings) {
|
||||
// _engine->_task->feedInput(timings);
|
||||
_engine->feedInput<WorkloadEngineBuilder::Inputs>(1, timings);
|
||||
}
|
||||
|
|
|
@ -43,4 +43,23 @@ public:
|
|||
workload::EnginePointer _engine;
|
||||
};
|
||||
|
||||
class ControlViewsConfig : public workload::Job::Config {
|
||||
Q_OBJECT
|
||||
public:
|
||||
|
||||
signals:
|
||||
void dirty();
|
||||
};
|
||||
|
||||
class ControlViews {
|
||||
public:
|
||||
using Config = ControlViewsConfig;
|
||||
using Input = workload::VaryingSet2<workload::Views, workload::Timings>;
|
||||
using Output = workload::Views;
|
||||
using JobModel = workload::Job::ModelIO<ControlViews, Input, Output, Config>;
|
||||
|
||||
void configure(const Config& config) {}
|
||||
void run(const workload::WorkloadContextPointer& renderContext, const Input& inputs, Output& outputs);
|
||||
};
|
||||
|
||||
#endif // hifi_GameWorkload_h
|
||||
|
|
|
@ -248,7 +248,7 @@ void GameWorkloadRenderItem::render(RenderArgs* args) {
|
|||
batch.setUniformBuffer(0, getDrawViewBuffer());
|
||||
static const int NUM_VERTICES_PER_DRAWVIEWVERT = 2;
|
||||
static const int NUM_REGIONS = 3;
|
||||
batch.draw(gpu::TRIANGLE_STRIP, NUM_REGIONS * NUM_VERTICES_PER_DRAWVIEWVERT * _numDrawViewVerts, 0);
|
||||
batch.draw(gpu::TRIANGLE_STRIP, NUM_REGIONS * NUM_VERTICES_PER_DRAWVIEWVERT * _numDrawViewVerts * _numAllViews, 0);
|
||||
|
||||
}
|
||||
|
||||
|
|
|
@ -415,7 +415,10 @@ void EntityTreeRenderer::update(bool simulate) {
|
|||
EntityTreePointer tree = std::static_pointer_cast<EntityTree>(_tree);
|
||||
|
||||
// here we update _currentFrame and _lastAnimated and sync with the server properties.
|
||||
tree->update(simulate);
|
||||
{
|
||||
PerformanceTimer perfTimer("tree::update");
|
||||
tree->update(simulate);
|
||||
}
|
||||
|
||||
{ // Update the rendereable entities as needed
|
||||
PROFILE_RANGE(simulation_physics, "Scene");
|
||||
|
@ -430,19 +433,22 @@ void EntityTreeRenderer::update(bool simulate) {
|
|||
scene->enqueueTransaction(transaction);
|
||||
}
|
||||
}
|
||||
workload::Transaction spaceTransaction;
|
||||
{ // update proxies in the workload::Space
|
||||
std::unique_lock<std::mutex> lock(_spaceLock);
|
||||
spaceTransaction.update(_spaceUpdates);
|
||||
_spaceUpdates.clear();
|
||||
}
|
||||
{
|
||||
std::vector<int32_t> staleProxies;
|
||||
tree->swapStaleProxies(staleProxies);
|
||||
spaceTransaction.remove(staleProxies);
|
||||
{
|
||||
PerformanceTimer perfTimer("workload::transaction");
|
||||
workload::Transaction spaceTransaction;
|
||||
{ // update proxies in the workload::Space
|
||||
std::unique_lock<std::mutex> lock(_spaceLock);
|
||||
_space->enqueueTransaction(spaceTransaction);
|
||||
spaceTransaction.update(_spaceUpdates);
|
||||
_spaceUpdates.clear();
|
||||
}
|
||||
{
|
||||
std::vector<int32_t> staleProxies;
|
||||
tree->swapStaleProxies(staleProxies);
|
||||
spaceTransaction.remove(staleProxies);
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(_spaceLock);
|
||||
_space->enqueueTransaction(spaceTransaction);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -31,6 +31,7 @@ void EntitySimulation::setEntityTree(EntityTreePointer tree) {
|
|||
void EntitySimulation::updateEntities() {
|
||||
QMutexLocker lock(&_mutex);
|
||||
uint64_t now = usecTimestampNow();
|
||||
PerformanceTimer perfTimer("EntitySimulation::updateEntities");
|
||||
|
||||
// these methods may accumulate entries in _entitiesToBeDeleted
|
||||
expireMortalEntities(now);
|
||||
|
|
|
@ -1819,6 +1819,7 @@ void EntityTree::addToNeedsParentFixupList(EntityItemPointer entity) {
|
|||
|
||||
void EntityTree::update(bool simulate) {
|
||||
PROFILE_RANGE(simulation_physics, "UpdateTree");
|
||||
PerformanceTimer perfTimer("UpdateTreen");
|
||||
withWriteLock([&] {
|
||||
fixupNeedsParentFixups();
|
||||
if (simulate && _simulation) {
|
||||
|
|
|
@ -187,6 +187,7 @@ public:
|
|||
|
||||
template <class I> void feedInput(const I& in) { _concept->editInput().template edit<I>() = in; }
|
||||
|
||||
template <class I, class S> void feedInput(int index, const S& inS) { (_concept->editInput().template editN<I>(index)).template edit<S>() = inS; }
|
||||
|
||||
QConfigPointer& getConfiguration() const { return _concept->getConfiguration(); }
|
||||
void applyConfiguration() { return _concept->applyConfiguration(); }
|
||||
|
@ -378,8 +379,33 @@ public:
|
|||
|
||||
protected:
|
||||
};
|
||||
}
|
||||
|
||||
template <class JC>
|
||||
class Engine : public Task<JC> {
|
||||
public:
|
||||
using Context = JC;
|
||||
using ContextPointer = std::shared_ptr<Context>;
|
||||
using Config = TaskConfig;
|
||||
using TaskType = Task<JC>;
|
||||
using ConceptPointer = typename TaskType::ConceptPointer;
|
||||
|
||||
Engine(ConceptPointer concept) : TaskType(concept) {}
|
||||
~Engine() = default;
|
||||
|
||||
void reset(const ContextPointer& context) {
|
||||
_context = context;
|
||||
}
|
||||
|
||||
void run() {
|
||||
if (_context) {
|
||||
TaskType::run(_context);
|
||||
}
|
||||
}
|
||||
|
||||
protected:
|
||||
ContextPointer _context;
|
||||
};
|
||||
}
|
||||
|
||||
#define Task_DeclareTypeAliases(ContextType) \
|
||||
using JobConfig = task::JobConfig; \
|
||||
|
@ -387,6 +413,7 @@ protected:
|
|||
template <class T> using PersistentConfig = task::PersistentConfig<T>; \
|
||||
using Job = task::Job<ContextType>; \
|
||||
using Task = task::Task<ContextType>; \
|
||||
using _Engine = task::Engine<ContextType>; \
|
||||
using Varying = task::Varying; \
|
||||
template < typename T0, typename T1 > using VaryingSet2 = task::VaryingSet2<T0, T1>; \
|
||||
template < typename T0, typename T1, typename T2 > using VaryingSet3 = task::VaryingSet3<T0, T1, T2>; \
|
||||
|
|
|
@ -15,9 +15,5 @@
|
|||
|
||||
namespace workload {
|
||||
WorkloadContext::WorkloadContext(const SpacePointer& space) : task::JobContext(trace_workload()), _space(space) {}
|
||||
|
||||
void Engine::reset(const WorkloadContextPointer& context) {
|
||||
_context = context;
|
||||
}
|
||||
} // namespace workload
|
||||
|
||||
|
|
|
@ -33,26 +33,7 @@ namespace workload {
|
|||
|
||||
using WorkloadContextPointer = std::shared_ptr<WorkloadContext>;
|
||||
Task_DeclareTypeAliases(WorkloadContext)
|
||||
|
||||
class Engine {
|
||||
public:
|
||||
Engine(const std::shared_ptr<Task>& task) : _task(task), _context(nullptr) {
|
||||
}
|
||||
~Engine() = default;
|
||||
|
||||
void reset(const WorkloadContextPointer& context);
|
||||
|
||||
void run() { if (_context) { _task->run(_context); } }
|
||||
|
||||
std::shared_ptr<TaskConfig> getConfiguration() { return _task->getConfiguration(); }
|
||||
|
||||
std::shared_ptr<Task> _task;
|
||||
|
||||
protected:
|
||||
|
||||
private:
|
||||
WorkloadContextPointer _context;
|
||||
};
|
||||
using Engine = _Engine;
|
||||
using EnginePointer = std::shared_ptr<Engine>;
|
||||
} // namespace workload
|
||||
|
||||
|
|
|
@ -24,7 +24,7 @@ void PerformSpaceTransaction::run(const WorkloadContextPointer& context) {
|
|||
}
|
||||
|
||||
void SpaceClassifierTask::build(JobModel& model, const Varying& in, Varying& out) {
|
||||
model.addJob<SetupViews>("setupViews", in);
|
||||
model.addJob<AssignSpaceViews >("assignSpaceViews", in);
|
||||
model.addJob<PerformSpaceTransaction>("updateSpace");
|
||||
const auto regionTrackerOut = model.addJob<RegionTracker>("regionTracker");
|
||||
const auto regionChanges = regionTrackerOut.getN<RegionTracker::Outputs>(1);
|
||||
|
|
|
@ -13,52 +13,59 @@
|
|||
using namespace workload;
|
||||
|
||||
|
||||
void AssignSpaceViews::run(const WorkloadContextPointer& renderContext, const Input& inputs) {
|
||||
// Just do what it says
|
||||
renderContext->_space->setViews(inputs);
|
||||
}
|
||||
|
||||
void SetupViews::configure(const Config& config) {
|
||||
data = config.data;
|
||||
}
|
||||
|
||||
void SetupViews::run(const WorkloadContextPointer& renderContext, const Input& inputs) {
|
||||
void SetupViews::run(const WorkloadContextPointer& renderContext, const Input& inputs, Output& outputs) {
|
||||
// If views are frozen don't use the input
|
||||
if (!data.freezeViews) {
|
||||
_views = inputs;
|
||||
}
|
||||
|
||||
auto& outViews = outputs;
|
||||
outViews.clear();
|
||||
|
||||
// Filter the first view centerer on the avatar head if needed
|
||||
Views usedViews;
|
||||
if (_views.size() >= 2) {
|
||||
if (data.useAvatarView) {
|
||||
usedViews.push_back(_views[0]);
|
||||
usedViews.insert(usedViews.end(), _views.begin() + 2, _views.end());
|
||||
outViews.push_back(_views[0]);
|
||||
outViews.insert(outViews.end(), _views.begin() + 2, _views.end());
|
||||
} else {
|
||||
usedViews.insert(usedViews.end(), _views.begin() + 1, _views.end());
|
||||
outViews.insert(outViews.end(), _views.begin() + 1, _views.end());
|
||||
}
|
||||
} else {
|
||||
usedViews = _views;
|
||||
outViews = _views;
|
||||
}
|
||||
|
||||
// Force frutum orientation horizontal if needed
|
||||
if (usedViews.size() > 0 && data.forceViewHorizontal) {
|
||||
usedViews[0].makeHorizontal();
|
||||
if (outViews.size() > 0 && data.forceViewHorizontal) {
|
||||
outViews[0].makeHorizontal();
|
||||
}
|
||||
|
||||
// Force frutum orientation horizontal if needed
|
||||
if (usedViews.size() > 0 && data.simulateSecondaryCamera) {
|
||||
auto view = usedViews[0];
|
||||
if (outViews.size() > 0 && data.simulateSecondaryCamera) {
|
||||
auto view = outViews[0];
|
||||
auto secondaryDirectionFlat = glm::normalize(glm::vec3(view.direction.x, 0.0f, view.direction.z));
|
||||
auto secondaryDirection = glm::normalize(glm::vec3(secondaryDirectionFlat.z, 0.0f, -secondaryDirectionFlat.x));
|
||||
|
||||
view.origin += -30.0f * secondaryDirection;
|
||||
view.direction = -secondaryDirectionFlat;
|
||||
view.origin += -20.0f * secondaryDirection;
|
||||
view.direction = -secondaryDirection;
|
||||
|
||||
usedViews.insert(usedViews.begin() + 1, view);
|
||||
outViews.insert(outViews.begin() + 1, view);
|
||||
}
|
||||
|
||||
// Update regions based on the current config
|
||||
for (auto& v : usedViews) {
|
||||
for (auto& v : outViews) {
|
||||
View::updateRegions(v, (float*) &data);
|
||||
}
|
||||
|
||||
// Views are setup, assign to the Space
|
||||
renderContext->_space->setViews(usedViews);
|
||||
// outViews is ready to be used
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -78,16 +78,25 @@ namespace workload {
|
|||
public:
|
||||
using Config = SetupViewsConfig;
|
||||
using Input = Views;
|
||||
using JobModel = Job::ModelI<SetupViews, Input, Config>;
|
||||
using Output = Views;
|
||||
using JobModel = Job::ModelIO<SetupViews, Input, Output, Config>;
|
||||
|
||||
void configure(const Config& config);
|
||||
void run(const workload::WorkloadContextPointer& renderContext, const Input& inputs);
|
||||
void run(const workload::WorkloadContextPointer& renderContext, const Input& inputs, Output& outputs);
|
||||
|
||||
protected:
|
||||
Config::Data data;
|
||||
Views _views;
|
||||
};
|
||||
|
||||
class AssignSpaceViews {
|
||||
public:
|
||||
using Input = Views;
|
||||
using JobModel = Job::ModelI<AssignSpaceViews, Input>;
|
||||
|
||||
void run(const workload::WorkloadContextPointer& renderContext, const Input& inputs);
|
||||
};
|
||||
|
||||
} // namespace workload
|
||||
|
||||
#endif // hifi_workload_ViewTask_h
|
Loading…
Reference in a new issue