// // KRScene.cpp // KREngine // // Copyright 2012 Kearwood Gilbert. All rights reserved. // // Redistribution and use in source and binary forms, with or without modification, are // permitted provided that the following conditions are met: // // 1. Redistributions of source code must retain the above copyright notice, this list of // conditions and the following disclaimer. // // 2. Redistributions in binary form must reproduce the above copyright notice, this list // of conditions and the following disclaimer in the documentation and/or other materials // provided with the distribution. // // THIS SOFTWARE IS PROVIDED BY KEARWOOD GILBERT ''AS IS'' AND ANY EXPRESS OR IMPLIED // WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND // FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL KEARWOOD GILBERT OR // CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON // ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING // NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF // ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // // The views and conclusions contained in the software and documentation are those of the // authors and should not be interpreted as representing official policies, either expressed // or implied, of Kearwood Gilbert. // #include #import "KRVector3.h" #import "KRMat4.h" #import "tinyxml2.h" #import "KRLight.h" #import "KRScene.h" #import "KRNode.h" #import "KRStockGeometry.h" #import "KRDirectionalLight.h" #import "KRSpotLight.h" #import "KRPointLight.h" #import "KRQuaternion.h" const long KRENGINE_OCCLUSION_TEST_EXPIRY = 60; KRScene::KRScene(KRContext &context, std::string name) : KRResource(context, name) { m_pFirstLight = NULL; m_pRootNode = new KRNode(*this, "scene_root"); m_skyBoxName = ""; } KRScene::~KRScene() { delete m_pRootNode; m_pRootNode = NULL; } #if TARGET_OS_IPHONE void KRScene::renderFrame(float deltaTime) { getContext().startFrame(deltaTime); KRCamera *camera = find(); if(camera == NULL) { // Add a default camera if none are present camera = new KRCamera(*this, "default_camera"); m_pRootNode->addChild(camera); } camera->renderFrame(deltaTime); getContext().endFrame(deltaTime); } void KRScene::render(KRCamera *pCamera, std::map &visibleBounds, const KRViewport &viewport, KRNode::RenderPass renderPass, bool new_frame) { if(new_frame) { // Expire cached occlusion test results. // Cached "failed" results are expired on the next frame (marked with .second of -1) // Cached "success" results are expired after KRENGINE_OCCLUSION_TEST_EXPIRY frames (marked with .second of the last frame std::set expired_visible_bounds; for(std::map::iterator visible_bounds_itr = visibleBounds.begin(); visible_bounds_itr != visibleBounds.end(); visible_bounds_itr++) { if((*visible_bounds_itr).second == -1 || (*visible_bounds_itr).second + KRENGINE_OCCLUSION_TEST_EXPIRY < getContext().getCurrentFrame()) { expired_visible_bounds.insert((*visible_bounds_itr).first); } } for(std::set::iterator expired_visible_bounds_itr = expired_visible_bounds.begin(); expired_visible_bounds_itr != expired_visible_bounds.end(); expired_visible_bounds_itr++) { visibleBounds.erase(*expired_visible_bounds_itr); } } if(getFirstLight() == NULL) { addDefaultLights(); } std::vector lights; updateOctree(); pCamera->settings.setSkyBox(m_skyBoxName); // This is temporary until the camera is moved into the scene graph std::set outerNodes = std::set(m_nodeTree.getOuterSceneNodes()); // HACK - Copying the std::set as it is potentially modified as KRNode's update their bounds during the iteration. This is very expensive and will be eliminated in the future. // Get lights from outer nodes (directional lights, which have no bounds) for(std::set::iterator itr=outerNodes.begin(); itr != outerNodes.end(); itr++) { KRNode *node = (*itr); KRLight *light = dynamic_cast(node); if(light) { lights.push_back(light); } } // Render outer nodes for(std::set::iterator itr=outerNodes.begin(); itr != outerNodes.end(); itr++) { KRNode *node = (*itr); node->render(pCamera, lights, viewport, renderPass); } std::vector remainingOctrees; std::vector remainingOctreesTestResults; std::vector remainingOctreesTestResultsOnly; if(m_nodeTree.getRootNode() != NULL) { remainingOctrees.push_back(m_nodeTree.getRootNode()); } std::vector newRemainingOctrees; std::vector newRemainingOctreesTestResults; while((!remainingOctrees.empty() || !remainingOctreesTestResults.empty())) { newRemainingOctrees.clear(); newRemainingOctreesTestResults.clear(); for(std::vector::iterator octree_itr = remainingOctrees.begin(); octree_itr != remainingOctrees.end(); octree_itr++) { render(*octree_itr, visibleBounds, pCamera, lights, viewport, renderPass, newRemainingOctrees, newRemainingOctreesTestResults, remainingOctreesTestResultsOnly, false, false); } for(std::vector::iterator octree_itr = remainingOctreesTestResults.begin(); octree_itr != remainingOctreesTestResults.end(); octree_itr++) { render(*octree_itr, visibleBounds, pCamera, lights, viewport, renderPass, newRemainingOctrees, newRemainingOctreesTestResults, remainingOctreesTestResultsOnly, true, false); } remainingOctrees = newRemainingOctrees; remainingOctreesTestResults = newRemainingOctreesTestResults; } newRemainingOctrees.clear(); newRemainingOctreesTestResults.clear(); for(std::vector::iterator octree_itr = remainingOctreesTestResultsOnly.begin(); octree_itr != remainingOctreesTestResultsOnly.end(); octree_itr++) { render(*octree_itr, visibleBounds, pCamera, lights, viewport, renderPass, newRemainingOctrees, newRemainingOctreesTestResults, remainingOctreesTestResultsOnly, true, true); } } void KRScene::render(KROctreeNode *pOctreeNode, std::map &visibleBounds, KRCamera *pCamera, std::vector lights, const KRViewport &viewport, KRNode::RenderPass renderPass, std::vector &remainingOctrees, std::vector &remainingOctreesTestResults, std::vector &remainingOctreesTestResultsOnly, bool bOcclusionResultsPass, bool bOcclusionTestResultsOnly) { if(pOctreeNode) { KRAABB octreeBounds = pOctreeNode->getBounds(); if(bOcclusionResultsPass) { // ----====---- Occlusion results pass ----====---- if(pOctreeNode->m_occlusionTested) { GLuint params = 0; GLDEBUG(glGetQueryObjectuivEXT(pOctreeNode->m_occlusionQuery, GL_QUERY_RESULT_EXT, ¶ms)); if(params) { // Record the frame number that the test has passed on visibleBounds[octreeBounds] = getContext().getCurrentFrame(); if(!bOcclusionTestResultsOnly) { // Schedule a pass to perform the rendering remainingOctrees.push_back(pOctreeNode); } } else { // Record -1 to indicate that the visibility test had failed visibleBounds[octreeBounds] = -1; } GLDEBUG(glDeleteQueriesEXT(1, &pOctreeNode->m_occlusionQuery)); pOctreeNode->m_occlusionTested = false; pOctreeNode->m_occlusionQuery = 0; } } else { float min_coverage = 0.0f; float lod_coverage = pOctreeNode->getBounds().coverage(viewport.getViewProjectionMatrix(), viewport.getSize()); // Cull against the view frustrum if(lod_coverage > min_coverage) { // ----====---- Rendering and occlusion test pass ----====---- bool bVisible = false; bool bNeedOcclusionTest = true; if(!bVisible) { // Assume bounding boxes are visible without occlusion test queries if the camera is inside the box. // The near clipping plane of the camera is taken into consideration by expanding the match area KRAABB cameraExtents = KRAABB(viewport.getCameraPosition() - KRVector3(pCamera->settings.getPerspectiveNearZ()), viewport.getCameraPosition() + KRVector3(pCamera->settings.getPerspectiveNearZ())); bVisible = octreeBounds.intersects(cameraExtents); if(bVisible) { // Record the frame number in which the camera was within the bounds visibleBounds[octreeBounds] = getContext().getCurrentFrame(); bNeedOcclusionTest = false; } } if(!bVisible) { // Check if a previous occlusion query has returned true, taking advantage of temporal consistency of visible elements from frame to frame // If the previous frame rendered this octree, then attempt to render it in this frame without performing a pre-occlusion test std::map::iterator match_itr = visibleBounds.find(octreeBounds); if(match_itr != visibleBounds.end()) { if((*match_itr).second == -1) { // We have already tested these bounds with a negative result bNeedOcclusionTest = false; } else { bVisible = true; // We set bNeedOcclusionTest to false only when the previous occlusion test is old and we need to perform an occlusion test to record if this octree node was visible for the next frame bNeedOcclusionTest = false; } } } if(!bVisible && bNeedOcclusionTest) { // Optimization: If this is an empty octree node with only a single child node, then immediately try to render the child node without an occlusion test for this higher level, as it would be more expensive than the occlusion test for the child if(pOctreeNode->getSceneNodes().empty()) { int child_count = 0; for(int i=0; i<8; i++) { if(pOctreeNode->getChildren()[i] != NULL) child_count++; } if(child_count == 1) bVisible = true; } } if(bNeedOcclusionTest) { pOctreeNode->beginOcclusionQuery(); KRMat4 matModel = KRMat4(); matModel.scale(octreeBounds.size() / 2.0f); matModel.translate(octreeBounds.center()); KRMat4 mvpmatrix = matModel * viewport.getViewProjectionMatrix(); getContext().getModelManager()->bindVBO((void *)KRENGINE_VBO_3D_CUBE, KRENGINE_VBO_3D_CUBE_SIZE, true, false, false, false, false, false, false); // Enable additive blending if(renderPass != KRNode::RENDER_PASS_FORWARD_TRANSPARENT && renderPass != KRNode::RENDER_PASS_ADDITIVE_PARTICLES && renderPass != KRNode::RENDER_PASS_VOLUMETRIC_EFFECTS_ADDITIVE) { GLDEBUG(glEnable(GL_BLEND)); } GLDEBUG(glBlendFunc(GL_ONE, GL_ONE)); if(renderPass == KRNode::RENDER_PASS_FORWARD_OPAQUE || renderPass == KRNode::RENDER_PASS_DEFERRED_GBUFFER || renderPass == KRNode::RENDER_PASS_DEFERRED_OPAQUE || renderPass == KRNode::RENDER_PASS_SHADOWMAP) { // Disable z-buffer write GLDEBUG(glDepthMask(GL_FALSE)); } if(getContext().getShaderManager()->selectShader("occlusion_test", *pCamera, lights, 0, viewport, matModel, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, KRNode::RENDER_PASS_FORWARD_TRANSPARENT)) { GLDEBUG(glDrawArrays(GL_TRIANGLE_STRIP, 0, 14)); } if(renderPass == KRNode::RENDER_PASS_FORWARD_OPAQUE || renderPass == KRNode::RENDER_PASS_DEFERRED_GBUFFER || renderPass == KRNode::RENDER_PASS_DEFERRED_OPAQUE || renderPass == KRNode::RENDER_PASS_SHADOWMAP) { // Re-enable z-buffer write GLDEBUG(glDepthMask(GL_TRUE)); } pOctreeNode->endOcclusionQuery(); if(renderPass != KRNode::RENDER_PASS_FORWARD_TRANSPARENT && renderPass != KRNode::RENDER_PASS_ADDITIVE_PARTICLES && renderPass != KRNode::RENDER_PASS_VOLUMETRIC_EFFECTS_ADDITIVE) { GLDEBUG(glDisable(GL_BLEND)); } else if(renderPass == KRNode::RENDER_PASS_FORWARD_TRANSPARENT) { GLDEBUG(glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA)); } else { GLDEBUG(glBlendFunc(GL_ONE, GL_ONE)); // RENDER_PASS_FORWARD_TRANSPARENT and RENDER_PASS_VOLUMETRIC_EFFECTS_ADDITIVE } if(bVisible) { // Schedule a pass to get the result of the occlusion test only for future frames and passes, without rendering the model or recurring further remainingOctreesTestResultsOnly.push_back(pOctreeNode); } else { // Schedule a pass to get the result of the occlusion test and continue recursion and rendering if test is true remainingOctreesTestResults.push_back(pOctreeNode); } } if(bVisible) { // Add lights that influence this octree level and its children to the stack int light_count = 0; for(std::set::iterator itr=pOctreeNode->getSceneNodes().begin(); itr != pOctreeNode->getSceneNodes().end(); itr++) { KRNode *node = (*itr); KRLight *light = dynamic_cast(node); if(light) { lights.push_back(light); light_count++; } } // Render objects that are at this octree level for(std::set::iterator itr=pOctreeNode->getSceneNodes().begin(); itr != pOctreeNode->getSceneNodes().end(); itr++) { //assert(pOctreeNode->getBounds().contains((*itr)->getBounds())); // Sanity check (*itr)->render(pCamera, lights, viewport, renderPass); } // Render child octrees const int *childOctreeOrder = renderPass == KRNode::RENDER_PASS_FORWARD_TRANSPARENT || renderPass == KRNode::RENDER_PASS_ADDITIVE_PARTICLES || renderPass == KRNode::RENDER_PASS_VOLUMETRIC_EFFECTS_ADDITIVE ? viewport.getBackToFrontOrder() : viewport.getFrontToBackOrder(); for(int i=0; i<8; i++) { render(pOctreeNode->getChildren()[childOctreeOrder[i]], visibleBounds, pCamera, lights, viewport, renderPass, remainingOctrees, remainingOctreesTestResults, remainingOctreesTestResultsOnly, false, false); } // Remove lights added at this octree level from the stack while(light_count--) { lights.pop_back(); } } } } } // fprintf(stderr, "Octree culled: (%f, %f, %f) - (%f, %f, %f)\n", pOctreeNode->getBounds().min.x, pOctreeNode->getBounds().min.y, pOctreeNode->getBounds().min.z, pOctreeNode->getBounds().max.x, pOctreeNode->getBounds().max.y, pOctreeNode->getBounds().max.z); } #endif std::string KRScene::getExtension() { return "krscene"; } KRNode *KRScene::getRootNode() { return m_pRootNode; } bool KRScene::save(KRDataBlock &data) { tinyxml2::XMLDocument doc; tinyxml2::XMLElement *scene_node = doc.NewElement( "scene" ); doc.InsertEndChild(scene_node); m_pRootNode->saveXML(scene_node); scene_node->SetAttribute("skybox", m_skyBoxName.c_str()); // This is temporary until the camera is moved into the scene graph tinyxml2::XMLPrinter p; doc.Print(&p); data.append((void *)p.CStr(), strlen(p.CStr())+1); return true; } KRScene *KRScene::Load(KRContext &context, const std::string &name, KRDataBlock *data) { data->append((void *)"\0", 1); // Ensure data is null terminated, to read as a string safely tinyxml2::XMLDocument doc; doc.Parse((char *)data->getStart()); KRScene *new_scene = new KRScene(context, name); tinyxml2::XMLElement *scene_element = doc.RootElement(); const char *szSkyBoxName = scene_element->Attribute("skybox"); new_scene->m_skyBoxName = szSkyBoxName ? szSkyBoxName : ""; // This is temporary until the camera is moved into the scene graph KRNode *n = KRNode::LoadXML(*new_scene, scene_element->FirstChildElement()); if(n) { new_scene->getRootNode()->addChild(n); } delete data; return new_scene; } KRLight *KRScene::getFirstLight() { if(m_pFirstLight == NULL) { m_pFirstLight = find(); } return m_pFirstLight; } void KRScene::notify_sceneGraphCreate(KRNode *pNode) { m_nodeTree.add(pNode); if(pNode->hasPhysics()) { m_physicsNodes.erase(pNode); } // m_newNodes.insert(pNode); } void KRScene::notify_sceneGraphDelete(KRNode *pNode) { m_nodeTree.remove(pNode); m_physicsNodes.erase(pNode); // // m_modifiedNodes.erase(pNode); // if(!m_newNodes.erase(pNode)) { // m_nodeTree.remove(pNode); // } } void KRScene::notify_sceneGraphModify(KRNode *pNode) { m_nodeTree.update(pNode); // m_modifiedNodes.insert(pNode); } void KRScene::updateOctree() { // for(std::set::iterator itr=m_newNodes.begin(); itr != m_newNodes.end(); itr++) { // m_nodeTree.add(*itr); // } // for(std::set::iterator itr=m_modifiedNodes.begin(); itr != m_modifiedNodes.end(); itr++) { // m_nodeTree.update(*itr); // } // m_newNodes.clear(); // m_modifiedNodes.clear(); } void KRScene::physicsUpdate(float deltaTime) { for(std::set::iterator itr=m_physicsNodes.begin(); itr != m_physicsNodes.end(); itr++) { (*itr)->physicsUpdate(deltaTime); } } void KRScene::addDefaultLights() { KRDirectionalLight *light1 = new KRDirectionalLight(*this, "default_light1"); light1->setLocalRotation((KRQuaternion(KRVector3(0.0, M_PI * 0.10, 0.0)) * KRQuaternion(KRVector3(0.0, 0.0, -M_PI * 0.15))).eulerXYZ()); m_pRootNode->addChild(light1); } KRAABB KRScene::getRootOctreeBounds() { if(m_nodeTree.getRootNode()) { return m_nodeTree.getRootNode()->getBounds(); } else { return KRAABB(-KRVector3::One(), KRVector3::One()); } } bool KRScene::lineCast(const KRVector3 &v0, const KRVector3 &v1, KRHitInfo &hitinfo) { return m_nodeTree.lineCast(v0, v1, hitinfo); } bool KRScene::rayCast(const KRVector3 &v0, const KRVector3 &dir, KRHitInfo &hitinfo) { return m_nodeTree.rayCast(v0, dir, hitinfo); }