Skip to content
Snippets Groups Projects
Commit e48d3e28 authored by Joseph Walton-Rivers's avatar Joseph Walton-Rivers
Browse files

add service-based physics

parent 7a24c12c
No related branches found
No related tags found
No related merge requests found
Showing
with 241 additions and 84 deletions
...@@ -86,6 +86,4 @@ prefabs: ...@@ -86,6 +86,4 @@ prefabs:
phys::Body: phys::Body:
type: kinematic type: kinematic
shape: shape:
type: box type: box
phys::Callbacks: \ No newline at end of file
phys::Cache:
\ No newline at end of file
...@@ -120,14 +120,18 @@ namespace demo { ...@@ -120,14 +120,18 @@ namespace demo {
constexpr fggl::math::vec3 COLOUR_BLUE{0.0F, 0.0F, 1.0F}; constexpr fggl::math::vec3 COLOUR_BLUE{0.0F, 0.0F, 1.0F};
constexpr float MOVE_FORCE{3.0F}; constexpr float MOVE_FORCE{3.0F};
RollBall::RollBall(fggl::App &app) : Game(app) { RollBall::RollBall(fggl::App &app) : Game(app), m_phys(nullptr) {
} }
void RollBall::activate() { void RollBall::activate() {
Game::activate(); Game::activate();
fggl::debug::log(fggl::debug::Level::info, "RollBall::activate()"); fggl::debug::log(fggl::debug::Level::info, "RollBall::activate()");
// attach physics
auto* physService = m_owner.service<fggl::phys::PhysicsProvider>();
auto* entFactory = m_owner.service<fggl::entity::EntityFactory>();
m_phys = physService->create(&world(), entFactory);
auto* assetLoader = m_owner.service<fggl::assets::Loader>(); auto* assetLoader = m_owner.service<fggl::assets::Loader>();
assetLoader->load("rollball.yml", fggl::entity::PROTOTYPE_ASSET); assetLoader->load("rollball.yml", fggl::entity::PROTOTYPE_ASSET);
...@@ -150,6 +154,17 @@ namespace demo { ...@@ -150,6 +154,17 @@ namespace demo {
setup_environment(world(), m_owner.service<fggl::entity::EntityFactory>(), WORLD_SIZE, state); setup_environment(world(), m_owner.service<fggl::entity::EntityFactory>(), WORLD_SIZE, state);
} }
void RollBall::deactivate() {
// we need to clean up physics
if ( m_phys != nullptr ) {
delete m_phys;
m_phys = nullptr;
}
// now let the rest of the scene do its bit
Game::deactivate();
}
fggl::math::vec3 calc_move_vector(const fggl::input::Input& input) { fggl::math::vec3 calc_move_vector(const fggl::input::Input& input) {
constexpr fggl::math::vec3 forward = fggl::math::FORWARD; constexpr fggl::math::vec3 forward = fggl::math::FORWARD;
constexpr fggl::math::vec3 right = fggl::math::RIGHT; constexpr fggl::math::vec3 right = fggl::math::RIGHT;
...@@ -176,6 +191,7 @@ namespace demo { ...@@ -176,6 +191,7 @@ namespace demo {
void RollBall::update() { void RollBall::update() {
Game::update(); Game::update();
m_phys->step();
const float deltaTime = 1 / 60.0F; const float deltaTime = 1 / 60.0F;
......
...@@ -26,6 +26,7 @@ ...@@ -26,6 +26,7 @@
#define DEMO_ROLLBALL_HPP #define DEMO_ROLLBALL_HPP
#include "fggl/scenes/game.hpp" #include "fggl/scenes/game.hpp"
#include "fggl/phys/service.hpp"
namespace demo { namespace demo {
...@@ -54,6 +55,7 @@ namespace demo { ...@@ -54,6 +55,7 @@ namespace demo {
explicit RollBall(fggl::App& app); explicit RollBall(fggl::App& app);
void activate() override; void activate() override;
void deactivate() override;
void update() override; void update() override;
void render(fggl::gfx::Graphics& gfx) override; void render(fggl::gfx::Graphics& gfx) override;
...@@ -61,6 +63,7 @@ namespace demo { ...@@ -61,6 +63,7 @@ namespace demo {
private: private:
constexpr static fggl::math::vec3 HINT_COLOUR{0.5f, 0.0f, 0.0f}; constexpr static fggl::math::vec3 HINT_COLOUR{0.5f, 0.0f, 0.0f};
RollState state; RollState state;
fggl::phys::PhysicsEngine* m_phys;
fggl::math::vec3 cameraOffset = {-15.0F, 15.0F, 0.0F}; fggl::math::vec3 cameraOffset = {-15.0F, 15.0F, 0.0F};
void closestPickup(fggl::entity::EntityManager& world); void closestPickup(fggl::entity::EntityManager& world);
......
...@@ -32,7 +32,7 @@ namespace fggl::entity { ...@@ -32,7 +32,7 @@ namespace fggl::entity {
auto name = prefab["name"].as<fggl::util::GUID>(); auto name = prefab["name"].as<fggl::util::GUID>();
#ifndef NDEBUG #ifndef NDEBUG
debug::log(debug::Level::info, "found prefab: {}", fggl::util::guidToString(name) ); debug::info("found prefab: {}", fggl::util::guidToString(name) );
#endif #endif
// setup the components // setup the components
...@@ -43,6 +43,11 @@ namespace fggl::entity { ...@@ -43,6 +43,11 @@ namespace fggl::entity {
ComponentSpec compSpec{}; ComponentSpec compSpec{};
compSpec.config = compEntry.second; compSpec.config = compEntry.second;
entity.components[compId] = compSpec; entity.components[compId] = compSpec;
entity.ordering.push_back(compId);
#ifndef NDEBUG
debug::trace("{} has component {}", fggl::util::guidToString(name), fggl::util::guidToString(compId));
#endif
} }
factory->define( name, entity ); factory->define( name, entity );
......
...@@ -26,6 +26,8 @@ namespace fggl::entity { ...@@ -26,6 +26,8 @@ namespace fggl::entity {
transform.origin( spec.get<math::vec3>( "origin", math::VEC3_ZERO ) ); transform.origin( spec.get<math::vec3>( "origin", math::VEC3_ZERO ) );
transform.euler( spec.get<math::vec3>( "rotation", math::VEC3_ZERO ) ); transform.euler( spec.get<math::vec3>( "rotation", math::VEC3_ZERO ) );
transform.scale( spec.get<math::vec3>( "scale", math::VEC3_ONES ) ); transform.scale( spec.get<math::vec3>( "scale", math::VEC3_ONES ) );
debug::trace("created transform for entity {}", (uint64_t)entity);
} }
void install_component_factories(entity::EntityFactory* factory) { void install_component_factories(entity::EntityFactory* factory) {
......
...@@ -20,6 +20,8 @@ ...@@ -20,6 +20,8 @@
#define FGGL_ENTITY_ENTITY_HPP #define FGGL_ENTITY_ENTITY_HPP
#include <cstdint> #include <cstdint>
#include "fggl/debug/logging.hpp"
#include "fggl/vendor/entt.hpp" #include "fggl/vendor/entt.hpp"
namespace fggl::entity { namespace fggl::entity {
...@@ -43,6 +45,11 @@ namespace fggl::entity { ...@@ -43,6 +45,11 @@ namespace fggl::entity {
template<typename Component> template<typename Component>
Component& get(EntityID entity) { Component& get(EntityID entity) {
#ifndef NDEBUG
if ( !has<Component>(entity) ) {
debug::error("Entity {} has no component of type {}", (uint64_t)entity, typeid(Component).name());
}
#endif
return m_registry.get<Component>(entity); return m_registry.get<Component>(entity);
} }
......
...@@ -47,17 +47,19 @@ namespace fggl::entity { ...@@ -47,17 +47,19 @@ namespace fggl::entity {
// invoke each component factory as required // invoke each component factory as required
auto entity = manager.create(); auto entity = manager.create();
for (auto &[name, data] : prototype.components) { for ( auto& component : prototype.ordering ) {
try { try {
m_factories.at(name)(data, manager, entity); auto& data = prototype.components.at(component);
m_factories.at(component)(data, manager, entity);
} catch (std::out_of_range& ex) { } catch (std::out_of_range& ex) {
#ifndef NDEBUG #ifndef NDEBUG
debug::log(debug::Level::error, "EntityFactory: Unknown component factory type '{}'", fggl::util::guidToString(name)); debug::log(debug::Level::error, "EntityFactory: Unknown component factory type '{}'", fggl::util::guidToString(component));
#endif #endif
manager.destroy(entity); manager.destroy(entity);
return fggl::entity::INVALID; return fggl::entity::INVALID;
} }
} }
return entity; return entity;
} catch (std::out_of_range& ex) { } catch (std::out_of_range& ex) {
#ifndef NDEBUG #ifndef NDEBUG
......
...@@ -46,6 +46,7 @@ namespace fggl::entity { ...@@ -46,6 +46,7 @@ namespace fggl::entity {
}; };
struct EntitySpec { struct EntitySpec {
std::vector<util::GUID> ordering;
std::map<util::GUID, ComponentSpec> components; std::map<util::GUID, ComponentSpec> components;
}; };
......
/*
* This file is part of FGGL.
*
* FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any
* later version.
*
* FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License along with FGGL.
* If not, see <https://www.gnu.org/licenses/>.
*/
//
// Created by webpigeon on 01/08/22.
//
#ifndef FGGL_PHYS_SERVICE_HPP
#define FGGL_PHYS_SERVICE_HPP
#include "fggl/modules/module.hpp"
#include "fggl/entity/module.hpp"
#include "fggl/phys/types.hpp"
namespace fggl::phys {
class PhysicsProvider {
public:
constexpr static const modules::ModuleService service = modules::make_service("fggl::phys::service");
virtual PhysicsEngine* create(entity::EntityManager* entityManager, entity::EntityFactory* factory) = 0;
};
}
#endif //FGGL_PHYS_SERVICE_HPP
...@@ -35,5 +35,6 @@ else() ...@@ -35,5 +35,6 @@ else()
src/module.cpp src/module.cpp
src/simulation.cpp src/simulation.cpp
src/phys_draw.cpp src/phys_draw.cpp
src/service.cpp
) )
endif() endif()
...@@ -23,22 +23,21 @@ ...@@ -23,22 +23,21 @@
#include "fggl/entity/module.hpp" #include "fggl/entity/module.hpp"
#include "fggl/phys/types.hpp" #include "fggl/phys/types.hpp"
#include "fggl/phys/service.hpp"
#include "fggl/phys/bullet/types.hpp" #include "fggl/phys/bullet/types.hpp"
namespace fggl::phys::bullet { namespace fggl::phys::bullet {
constexpr modules::ModuleService BIND_BULLET = modules::make_service("bind_bullet");
constexpr util::GUID CONFIG_PHYS_BODY = util::make_guid("phys::body"); constexpr util::GUID CONFIG_PHYS_BODY = util::make_guid("phys::body");
struct Bullet { struct Bullet {
constexpr static const char* name = "fggl::phys::bullet"; constexpr static const char* name = "fggl::phys::bullet";
constexpr static const std::array<modules::ModuleService, 1> provides = { constexpr static const std::array<modules::ModuleService, 1> provides = {
BIND_BULLET phys::PhysicsProvider::service
}; };
constexpr static const std::array<modules::ModuleService, 1> depends = { constexpr static const std::array<modules::ModuleService, 1> depends = {
entity::EntityFactory::service entity::EntityFactory::service
}; };
static bool factory(modules::ModuleService name, modules::Services& serviceManager); static bool factory(modules::ModuleService name, modules::Services& serviceManager);
}; };
......
...@@ -34,6 +34,7 @@ ...@@ -34,6 +34,7 @@
#define FGGL_PHYS_BULLET_MOTION_HPP #define FGGL_PHYS_BULLET_MOTION_HPP
#include "types.hpp" #include "types.hpp"
#include "fggl/debug/logging.hpp"
namespace fggl::phys::bullet { namespace fggl::phys::bullet {
...@@ -41,11 +42,16 @@ namespace fggl::phys::bullet { ...@@ -41,11 +42,16 @@ namespace fggl::phys::bullet {
public: public:
FgglMotionState(entity::EntityManager* world, entity::EntityID entity) : m_world(world), m_entity(entity) { FgglMotionState(entity::EntityManager* world, entity::EntityID entity) : m_world(world), m_entity(entity) {
} }
virtual ~FgglMotionState() = default; ~FgglMotionState() override = default;
void getWorldTransform(btTransform& worldTrans) const override { void getWorldTransform(btTransform& worldTrans) const override {
const auto& transform = m_world->get<fggl::math::Transform>(m_entity); const auto* transform = m_world->tryGet<fggl::math::Transform>(m_entity);
worldTrans.setFromOpenGLMatrix( glm::value_ptr(transform.model()) ); if ( transform == nullptr ) {
debug::warning("BT: attempted to get transform of entity without transform component.");
return;
}
worldTrans.setFromOpenGLMatrix(glm::value_ptr(transform->model()));
} }
void setWorldTransform(const btTransform& worldTrans) override { void setWorldTransform(const btTransform& worldTrans) override {
......
/*
* This file is part of FGGL.
*
* FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any
* later version.
*
* FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License along with FGGL.
* If not, see <https://www.gnu.org/licenses/>.
*/
//
// Created by webpigeon on 01/08/22.
//
#ifndef FGGL_INTEGRATIONS_BULLET_PHYS_BULLET_BULLETSERVICE_H
#define FGGL_INTEGRATIONS_BULLET_PHYS_BULLET_BULLETSERVICE_H
#include "fggl/phys/service.hpp"
namespace fggl::phys::bullet {
class BulletProvider : public phys::PhysicsProvider {
public:
PhysicsEngine* create(entity::EntityManager* entityManager, entity::EntityFactory* factory) override;
};
}
#endif //FGGL_INTEGRATIONS_BULLET_PHYS_BULLET_BULLETSERVICE_H
...@@ -83,6 +83,8 @@ namespace fggl::phys::bullet { ...@@ -83,6 +83,8 @@ namespace fggl::phys::bullet {
std::vector<entity::EntityID> raycastAll(math::vec3 from, math::vec3 to) override; std::vector<entity::EntityID> raycastAll(math::vec3 from, math::vec3 to) override;
std::vector<entity::EntityID> sweep(PhyShape& shape, math::Transform& from, math::Transform& to) override; std::vector<entity::EntityID> sweep(PhyShape& shape, math::Transform& from, math::Transform& to) override;
void addBody(entity::EntityID entity, BulletBody& body);
bool debugDraw = false; bool debugDraw = false;
private: private:
entity::EntityManager* m_ecs; entity::EntityManager* m_ecs;
......
...@@ -17,76 +17,13 @@ ...@@ -17,76 +17,13 @@
// //
#include "fggl/phys/bullet/module.hpp" #include "fggl/phys/bullet/module.hpp"
#include "fggl/phys/bullet/motion.hpp" #include "fggl/phys/bullet/service.hpp"
namespace fggl::phys::bullet { namespace fggl::phys::bullet {
constexpr float UNIT_EXTENT = 0.5F;
btCollisionShape* create_bullet_shape(const YAML::Node& node) {
auto type = node["type"].as<std::string>("MISSING");
if ( type == "sphere" ) {
auto radius = node["radius"].as<float>(UNIT_EXTENT);
return new btSphereShape(radius);
} else if ( type == "box" ) {
auto extents = node["extents"].as<math::vec3>(math::vec3(UNIT_EXTENT));
return new btBoxShape(btVector3(extents.x, extents.y, extents.z));
} else {
debug::warning("unknown shape type: {}, assuming unit sphere", type);
return new btSphereShape(0.5F);
}
}
void setup_body(BodyType type, btRigidBody& body) {
if ( type != BodyType::STATIC ) {
body.setRestitution(0.5F);
body.setRollingFriction(0.0F);
// kinematic bodies cannot sleep
if (type == BodyType::KINEMATIC) {
auto flags = (body.getCollisionFlags() | btCollisionObject::CF_KINEMATIC_OBJECT);
body.setCollisionFlags(flags);
body.setActivationState(DISABLE_DEACTIVATION);
}
}
}
void add_bt_body(const entity::ComponentSpec& spec, entity::EntityManager& manager, const entity::EntityID& id) {
auto& body = manager.add<BulletBody>(id);
body.motion = new FgglMotionState(&manager, id);
auto type = spec.get<BodyType>("type", BodyType::DYNAMIC);
auto mass = 0.0F;
// non-static objects can have mass
if ( type != BodyType::STATIC ) {
mass = spec.get<float>("mass", 1.0F);
}
// calculate inertia
btCollisionShape* shape = create_bullet_shape(spec.config["shape"]);
btVector3 inertia(0, 0, 0);
if ( type != BodyType::STATIC ) {
shape->calculateLocalInertia(mass, inertia);
}
// construction information for the simulation
btRigidBody::btRigidBodyConstructionInfo constructionInfo {
mass,
body.motion,
shape,
inertia
};
body.body = new btRigidBody(constructionInfo);
setup_body(type, *body.body);
// add the body to the simulation
}
bool Bullet::factory(modules::ModuleService service, modules::Services& services) { bool Bullet::factory(modules::ModuleService service, modules::Services& services) {
if ( service == BIND_BULLET ) { if ( service == phys::PhysicsProvider::service ) {
auto* entityFactory = services.get<entity::EntityFactory>(); services.bind<phys::PhysicsProvider, BulletProvider>();
entityFactory->bind(CONFIG_PHYS_BODY, add_bt_body);
} }
return false; return false;
} }
......
/*
* This file is part of FGGL.
*
* FGGL is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any
* later version.
*
* FGGL is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License along with FGGL.
* If not, see <https://www.gnu.org/licenses/>.
*/
//
// Created by webpigeon on 01/08/22.
//
#include "fggl/phys/bullet/service.hpp"
#include "fggl/phys/bullet/types.hpp"
#include "fggl/phys/bullet/motion.hpp"
namespace fggl::phys::bullet {
constexpr float UNIT_EXTENT = 0.5F;
btCollisionShape* create_bullet_shape(const YAML::Node& node) {
auto type = node["type"].as<std::string>("MISSING");
if ( type == "sphere" ) {
auto radius = node["radius"].as<float>(UNIT_EXTENT);
return new btSphereShape(radius);
} else if ( type == "box" ) {
auto extents = node["extents"].as<math::vec3>(math::vec3(UNIT_EXTENT));
return new btBoxShape(btVector3(extents.x, extents.y, extents.z));
} else {
debug::warning("unknown shape type: {}, assuming unit sphere", type);
return new btSphereShape(0.5F);
}
}
void setup_body(BodyType type, btRigidBody& body) {
if ( type != BodyType::STATIC ) {
body.setRestitution(0.5F);
body.setRollingFriction(0.0F);
// kinematic bodies cannot sleep
if (type == BodyType::KINEMATIC) {
auto flags = (body.getCollisionFlags() | btCollisionObject::CF_KINEMATIC_OBJECT);
body.setCollisionFlags(flags);
body.setActivationState(DISABLE_DEACTIVATION);
}
}
}
void add_bt_body(BulletPhysicsEngine* engine, const entity::ComponentSpec& spec, entity::EntityManager& manager, const entity::EntityID& id) {
auto& body = manager.add<BulletBody>(id);
body.motion = new FgglMotionState(&manager, id);
auto type = spec.get<BodyType>("type", BodyType::DYNAMIC);
auto mass = 0.0F;
// non-static objects can have mass
if ( type != BodyType::STATIC ) {
mass = spec.get<float>("mass", 1.0F);
}
// calculate inertia
btCollisionShape* shape = create_bullet_shape(spec.config["shape"]);
btVector3 inertia(0, 0, 0);
if ( type != BodyType::STATIC ) {
shape->calculateLocalInertia(mass, inertia);
}
// construction information for the simulation
btRigidBody::btRigidBodyConstructionInfo constructionInfo {
mass,
body.motion,
shape,
inertia
};
body.body = new btRigidBody(constructionInfo);
setup_body(type, *body.body);
// add the body to the simulation
engine->addBody(id, body);
if ( type == BodyType::DYNAMIC ) {
manager.add<phys::Dynamics>(id);
}
}
PhysicsEngine* BulletProvider::create(entity::EntityManager* entityManager, entity::EntityFactory* factory) {
auto* engine = new BulletPhysicsEngine(entityManager);
// FIXME: this won't work across multiple scenes - reactive approach might be better...
factory->bind(util::make_guid("phys::Body"), [engine](const entity::ComponentSpec& config, entity::EntityManager& manager, const entity::EntityID& id) {
add_bt_body(engine, config, manager, id);
});
engine->debugDraw = true;
return engine;
}
} // namespace fggl::phys::bullet
\ No newline at end of file
...@@ -49,7 +49,7 @@ namespace fggl::phys::bullet { ...@@ -49,7 +49,7 @@ namespace fggl::phys::bullet {
m_debug = std::make_unique<debug::BulletDebugDrawList>(); m_debug = std::make_unique<debug::BulletDebugDrawList>();
m_world->setDebugDrawer( m_debug.get() ); m_world->setDebugDrawer( m_debug.get() );
m_debug->setDebugMode(1); m_debug->setDebugMode(btIDebugDraw::DBG_DrawWireframe);
// callbacks (for handling bullet -> ecs) // callbacks (for handling bullet -> ecs)
...@@ -58,7 +58,8 @@ namespace fggl::phys::bullet { ...@@ -58,7 +58,8 @@ namespace fggl::phys::bullet {
} }
void BulletPhysicsEngine::step() { void BulletPhysicsEngine::step() {
checkForPhys(); // this is now handled on creation
//checkForPhys();
auto dynamicEnts = m_ecs->find<phys::Dynamics, phys::bullet::BulletBody>(); auto dynamicEnts = m_ecs->find<phys::Dynamics, phys::bullet::BulletBody>();
for (auto& ent : dynamicEnts) { for (auto& ent : dynamicEnts) {
...@@ -82,10 +83,14 @@ namespace fggl::phys::bullet { ...@@ -82,10 +83,14 @@ namespace fggl::phys::bullet {
} }
} }
void BulletPhysicsEngine::addBody(entity::EntityID /*entity*/, BulletBody &body) {
m_world->addRigidBody(body.body);
}
inline btCollisionShape* shape_to_bullet(const phys::RigidBody& fgglBody) { inline btCollisionShape* shape_to_bullet(const phys::RigidBody& fgglBody) {
if ( fgglBody.shape == nullptr ) { if ( fgglBody.shape == nullptr ) {
// they forgot to put a shape, we'll assume a unit sphere to avoid crashes... // they forgot to put a shape, we'll assume a unit sphere to avoid crashes...
return new btSphereShape(0.5f); return new btSphereShape(0.5F);
} }
switch (fgglBody.shape->type) { switch (fgglBody.shape->type) {
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment