// --------------------------------------------------------------------------- // Includes for all the program files to see // --------------------------------------------------------------------------- #include #include #include using namespace std; #include // --------------------------------------------------------------------------- // Includes // --------------------------------------------------------------------------- #include #include #include // --------------------------------------------------------------------------- // Includes // --------------------------------------------------------------------------- #include #include #include #include "ViewCellsParser.h" #include "ViewCellsParserXerces.h" #include "Mesh.h" #include "VspBspTree.h" #include "ViewCellBsp.h" #include "ViewCellsManager.h" #include "GzFileInputSource.h" #include "OspTree.h" #include "VspTree.h" #include "KdTree.h" #include "BvHierarchy.h" #include "HierarchyManager.h" namespace GtpVisibilityPreprocessor { // --------------------------------------------------------------------------- // Local data // // doNamespaces // Indicates whether namespace processing should be enabled or not. // The default is no, but -n overrides that. // // doSchema // Indicates whether schema processing should be enabled or not. // The default is no, but -s overrides that. // // schemaFullChecking // Indicates whether full schema constraint checking should be enabled or not. // The default is no, but -s overrides that. // // valScheme // Indicates what validation scheme to use. It defaults to 'auto', but // can be set via the -v= command. // --------------------------------------------------------------------------- static bool doNamespaces = false; static bool doSchema = false; static bool schemaFullChecking = false; static SAXParser::ValSchemes valScheme = SAXParser::Val_Auto; inline static bool ilt(Intersectable *obj1, Intersectable *obj2) { return obj1->mId < obj2->mId; } // --------------------------------------------------------------------------- // StdInParseHandlers: Constructors and Destructor // --------------------------------------------------------------------------- ViewCellsParseHandlers::ViewCellsParseHandlers(ObjectContainer *objects, BoundingBoxConverter *bconverter): mElementCount(0) , mAttrCount(0) , mCharacterCount(0) , mSpaceCount(0) , mViewCellsManager(NULL) , mVspBspTree(NULL) , mBspTree(NULL) , mViewCellsTree(NULL) , mCurrentState(PARSE_OPTIONS) , mCurrentViewCell(NULL) , mCurrentBspNode(NULL) , mCurrentVspNode(NULL) , mCurrentOspNode(NULL) , mCurrentBvhNode(NULL) , mObjects(objects) , mBoundingBoxConverter(bconverter) , mHierarchyManager(NULL) { std::stable_sort(mObjects->begin(), mObjects->end(), ilt); } ViewCellsParseHandlers::~ViewCellsParseHandlers() { } // --------------------------------------------------------------------------- // StdInParseHandlers: Implementation of the SAX DocumentHandler interface // --------------------------------------------------------------------------- void ViewCellsParseHandlers::endElement(const XMLCh* const name) { StrX lname(name); string element(lname.LocalForm()); if (element == "BoundingBoxes") { EndBoundingBoxes(); } if (element == "ViewCells") { EndViewCells(); } if (element == "ObjectSpaceHierarchy") { EndObjectSpaceHierarchy(); } // finished, create view cells manager if (element == "VisibilitySolution") { CreateViewCellsManager(); } if (element == "Interior") { switch (mCurrentState) { case PARSE_VIEWCELLS: EndViewCellInterior(); break; case PARSE_OBJECTSPACE_HIERARCHY: EndObjectSpaceHierarchyInterior(); break; case PARSE_VIEWSPACE_HIERARCHY: EndViewSpaceHierarchyInterior(); break; default: break; } } } void ViewCellsParseHandlers::EndObjectSpaceHierarchy() { if (mObjectSpaceHierarchyType == OSP) { // for a spatial subdivision, it is not necessary to store // the objects with the leaves, they can be classified now mHierarchyManager->mOspTree->InsertObjects( mHierarchyManager->mOspTree->mRoot, *mObjects); } } void ViewCellsParseHandlers::EndViewSpaceHierarchyInterior() { switch (mViewSpaceHierarchyType) { case BSP: EndBspInterior(); break; case VSP: EndVspInterior(); break; default: Debug << "not implemented view space hierarchy type " << mViewSpaceHierarchyType << endl; break; } } void ViewCellsParseHandlers::EndObjectSpaceHierarchyInterior() { switch (mObjectSpaceHierarchyType) { case OSP: EndOspInterior(); break; case BVH: EndBvhInterior(); break; default: Debug << "not implemented object space hierarchy type " << mViewSpaceHierarchyType << endl; break; } } void ViewCellsParseHandlers::EndBspInterior() { // go one up in the tree if (mCurrentBspNode->GetParent()) { cout << "]"; mCurrentBspNode = mCurrentBspNode->GetParent(); } } void ViewCellsParseHandlers::EndBvhInterior() { // go one up in the tree if (mCurrentBvhNode->GetParent()) { cout << "]"; mCurrentBvhNode = mCurrentBvhNode->GetParent(); } } void ViewCellsParseHandlers::EndOspInterior() { // go one up in the tree if (mCurrentOspNode->mParent) { cout << "]"; mCurrentOspNode = mCurrentOspNode->mParent; } } void ViewCellsParseHandlers::EndVspInterior() { // go one up in the tree if (mCurrentVspNode->GetParent()) { cout << "]"; mCurrentVspNode = mCurrentVspNode->GetParent(); } } void ViewCellsParseHandlers::EndViewCellInterior() { // go one up in the tree if (mCurrentViewCell->GetParent()) { cout << "]"; mCurrentViewCell = mCurrentViewCell->GetParent(); } } inline static bool vlt(ViewCell *v1, ViewCell *v2) { return v1->mId < v2->mId; } void ViewCellsParseHandlers::EndViewCells() { // sort view cells to help associating view cells according to their id //stable_sort(mViewCells.begin(), mViewCells.end(), vlt); // not parsing view cells anymore mCurrentState = PARSE_OPTIONS; } void ViewCellsParseHandlers::EndBoundingBoxes() { // all bounding boxes gathered in this step => // associate object ids with bounding boxes long startTime = GetTime(); if (mBoundingBoxConverter) mBoundingBoxConverter->IdentifyObjects(mIBoundingBoxes, *mObjects); Debug << "\nconverted bounding boxes to objects in " << TimeDiff(startTime, GetTime()) * 1e-3 << " secs" << endl; } void ViewCellsParseHandlers::StartBspElement(string element, AttributeList& attributes) { if (element == "Interior") { cout << "["; StartBspInterior(attributes); } if (element == "Leaf") { cout << "l"; StartBspLeaf(attributes); } } void ViewCellsParseHandlers::StartVspElement(string element, AttributeList& attributes) { if (element == "Interior") { cout << "["; StartVspInterior(attributes); } if (element == "Leaf") { cout << "l"; StartVspLeaf(attributes); } } void ViewCellsParseHandlers::StartOspElement(string element, AttributeList& attributes) { if (element == "Interior") { cout << "["; StartOspInterior(attributes); } if (element == "Leaf") { cout << "l"; StartOspLeaf(attributes); } } void ViewCellsParseHandlers::StartBvhElement(string element, AttributeList& attributes) { if (element == "Interior") { cout << "["; StartBvhInterior(attributes); } if (element == "Leaf") { cout << "l"; StartBvhLeaf(attributes); } } void ViewCellsParseHandlers::StartViewSpaceHierarchyElement(const std::string &element, AttributeList& attributes) { //-- use cell type according to the chosen method switch (mViewSpaceHierarchyType) { case BSP: StartBspElement(element, attributes); break; case VSP: StartVspElement(element, attributes); break; default: Debug << "not implemented" << endl; break; } } void ViewCellsParseHandlers::StartObjectSpaceHierarchyElement(const std::string &element, AttributeList& attributes) { //-- use cell type according to the chosen method switch (mObjectSpaceHierarchyType) { case OSP: StartOspElement(element, attributes); break; case BVH: StartBvhElement(element, attributes); break; default: Debug << "not implemented" << endl; break; } } void ViewCellsParseHandlers::StartViewCellHierarchyElement(const std::string &element, AttributeList& attributes) { // interiors + leaves interpreted view cells else if (element == "Interior") { cout << "["; StartViewCell(attributes, false); } if (element == "Leaf") { cout << "l"; StartViewCell(attributes, true); } } void ViewCellsParseHandlers::startElement(const XMLCh* const name, AttributeList& attributes) { StrX lname(name); string element(lname.LocalForm()); if (element == "ViewCells") { cout << "\nparsing view cells" << endl; mCurrentState = PARSE_VIEWCELLS; // create new view cells hierarchy mViewCellsTree = new ViewCellsTree(); } // decides about the view cell hierarchy if (element == "ViewSpaceHierarchy") { cout << "\nparsing view space hierarchy" << endl; mCurrentState = PARSE_VIEWSPACE_HIERARCHY; StartViewSpaceHierarchy(attributes); } // decides about the view cell hierarchy if (element == "ObjectSpaceHierarchy") { cout << "\nparsing object space hierarchy" << endl; mCurrentState = PARSE_OBJECTSPACE_HIERARCHY; StartObjectSpaceHierarchy(attributes); } // decides the used view cell hierarchy if (element == "BoundingBox") { cout << "b"; StartBoundingBox(attributes); } // parse view space hierarchy switch (mCurrentState) { case PARSE_VIEWSPACE_HIERARCHY: StartViewSpaceHierarchyElement(element, attributes); break; case PARSE_OBJECTSPACE_HIERARCHY: StartObjectSpaceHierarchyElement(element, attributes); break; case PARSE_VIEWCELLS: StartViewCellHierarchyElement(element, attributes); break; default: break; } ++ mElementCount; mAttrCount += attributes.getLength(); } void ViewCellsParseHandlers::StartViewCellPvs(ObjectPvs &pvs, const char *ptr) { // handle obect indices vector objIndices; char *endptr; while (1) { // read object ids const int index = strtol(ptr, &endptr, 10); if (ptr == endptr) break; objIndices.push_back(index); ptr = endptr; } // TODO: // 1) find objects and add them to pvs // 2) get view cell with specified id MeshInstance dummyInst(NULL); vector::const_iterator it, it_end = objIndices.end(); for (it = objIndices.begin(); it != it_end; ++ it) { const int objId = *it; dummyInst.SetId(objId); ObjectContainer::iterator oit = lower_bound(mObjects->begin(), mObjects->end(), (Intersectable *)&dummyInst, ilt); if ((oit != mObjects->end()) && ((*oit)->GetId() == objId)) { // $$JB we should store a float a per object which corresponds // to sumof pdfs, i.e. its relative visibility // temporarily set to 1.0f pvs.AddSample(*oit, 1.0f); } else { cout << "error: object with id " << objId << " does not exist" << endl; } } } void ViewCellsParseHandlers::StartViewSpaceHierarchy(AttributeList& attributes) { int len = attributes.getLength(); Vector3 bmin, bmax; for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); // hierarchy type if (attrName == "type") { if (strcmp(ptr, "bsp") == 0) { cout << "\nview space hierarchy: Bsp" << endl; mViewSpaceHierarchyType = BSP; } else if (strcmp(ptr, "vsp") == 0) { cout << "\nview space hierarchy: Vsp" << endl; mViewSpaceHierarchyType = VSP; } } else if (attrName == "min") // the view space extent { sscanf(ptr, "%f %f %f", &bmin.x, &bmin.y, &bmin.z); } else if (attrName == "max") { sscanf(ptr, "%f %f %f", &bmax.x, &bmax.y, &bmax.z); } } mViewSpaceBox = AxisAlignedBox3(bmin, bmax); // create the hierarchy based on this information CreateViewSpaceHierarchy(); } void ViewCellsParseHandlers::StartObjectSpaceHierarchy(AttributeList& attributes) { const int len = attributes.getLength(); Vector3 bmin, bmax; for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); // hierarchy type if (attrName == "type") { if (strcmp(ptr, "osp") == 0) { cout << "\nobject space hierarchy: Osp" << endl; mHierarchyManager = new HierarchyManager(HierarchyManager::KD_BASED_OBJ_SUBDIV); DEL_PTR(mHierarchyManager->mVspTree); mHierarchyManager->mVspTree = mVspTree; mObjectSpaceHierarchyType = OSP; //std::stable_sort(objects.begin(), objects.end(), ilt); mHierarchyManager->mOspTree->mBoundingBox.Initialize(); ObjectContainer::const_iterator oit, oit_end = mObjects->end(); //-- compute bounding box for (oit = mObjects->begin(); oit != oit_end; ++ oit) { Intersectable *obj = *oit; // compute bounding box of view space mHierarchyManager->mOspTree->mBoundingBox.Include(obj->GetBox()); } } else if (strcmp(ptr, "bvh") == 0) { cout << "\nobject space hierarchy: Bvh" << endl; mObjectSpaceHierarchyType = BVH; mHierarchyManager = new HierarchyManager(HierarchyManager::BV_BASED_OBJ_SUBDIV); DEL_PTR(mHierarchyManager->mVspTree); mHierarchyManager->mVspTree = mVspTree; } } } } void ViewCellsParseHandlers::StartBoundingBox(AttributeList& attributes) { int len = attributes.getLength(); Vector3 bmin, bmax; int id; for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); if (attrName == "id") { sscanf(ptr, "%d", &id); } if (attrName == "min") { sscanf(ptr, "%f %f %f", &bmin.x, &bmin.y, &bmin.z); } else if (attrName == "max") { sscanf(ptr, "%f %f %f", &bmax.x, &bmax.y, &bmax.z); } } AxisAlignedBox3 box(bmin, bmax); mIBoundingBoxes.push_back(IndexedBoundingBox(id, box)); //cout << "bbox: " << box << endl; } void ViewCellsParseHandlers::StartBspLeaf(AttributeList& attributes) { BspLeaf * leaf; if (mCurrentBspNode) // replace front or (if not NULL) back child { BspInterior *interior = dynamic_cast(mCurrentBspNode); leaf = new BspLeaf(interior); interior->ReplaceChildLink(NULL, leaf); } else { leaf = new BspLeaf(); mVspBspTree->mRoot = leaf; } /////////// //-- find associated view cell int viewCellId; int len = attributes.getLength(); for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); char *endptr = NULL; if (attrName == "viewCellId") { viewCellId = strtol(ptr, &endptr, 10); } } if (viewCellId >= 0) // valid view cell found { // TODO: get view cell with specified id ViewCellsMap::iterator vit = mViewCells.find(viewCellId); BspViewCell *viewCell = dynamic_cast((*vit).second); if (viewCell->GetId() == viewCellId) { // create new view cell for bsp nodes leaf->SetViewCell(viewCell); viewCell->mLeaves.push_back(leaf); } else { cout << "error: view cell does not exist" << endl; } } else { // add to invalid view space leaf->SetViewCell(mVspBspTree->GetOrCreateOutOfBoundsCell()); leaf->SetTreeValid(false); mVspBspTree->PropagateUpValidity(leaf); } } void ViewCellsParseHandlers::StartBspInterior(AttributeList& attributes) { Plane3 plane; int len = attributes.getLength(); for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); if (attrName == "plane") { sscanf(ptr, "%f %f %f %f", &plane.mNormal.x, &plane.mNormal.y, &plane.mNormal.z, &plane.mD); } } BspInterior* interior = new BspInterior(plane); if (mCurrentBspNode) // replace NULL child of parent with current node { BspInterior *parent = dynamic_cast(mCurrentBspNode); parent->ReplaceChildLink(NULL, interior); interior->SetParent(parent); } else { mVspBspTree->mRoot = interior; } mCurrentBspNode = interior; } ViewCell *ViewCellsParseHandlers::GetOrCreateViewCell(const int id, const bool isLeaf) { ViewCellsMap::iterator vit = mViewCells.find(id); if (vit != mViewCells.end()) { return (*vit).second; } else { ViewCell *vc; if (isLeaf) { vc = new ViewCellLeaf(); } else { vc = new ViewCellInterior(); } vc->SetId(id); mViewCells[id] = vc; return vc; } } void ViewCellsParseHandlers::StartViewCell(AttributeList& attributes, const bool isLeaf) { ViewCell *viewCell = NULL; const int len = attributes.getLength(); float mergeCost; ObjectPvs pvs; for (int i = 0; i < len; ++ i) { const string attrName(StrX(attributes.getName(i)).LocalForm()); if (attrName == "id") { const StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); char *endptr = NULL; const int id = strtol(ptr, &endptr, 10); // create new view cell, otherwise use reference. viewCell = GetOrCreateViewCell(id, isLeaf); if (mCurrentViewCell) // replace front or (if not NULL) back child { ViewCellInterior *interior = dynamic_cast(mCurrentViewCell); interior->SetupChildLink(viewCell); } else { // set the new root mViewCellsTree->SetRoot(viewCell); } if (!isLeaf) { mCurrentViewCell = viewCell; } } if (attrName == "pvs") { StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); // note: id must come before pvs! // otherwise view cell is undefined StartViewCellPvs(pvs, ptr); } else if (attrName == "active") { StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); char *endptr = NULL; const bool isActive = (bool)strtol(ptr, &endptr, 10); if (isActive) { // TODO } } else if (attrName == "mergecost") { StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); char *endptr = NULL; mergeCost = (float)strtod(ptr, &endptr); } } viewCell->SetMergeCost(mergeCost); viewCell->SetPvs(pvs); } void ViewCellsParseHandlers::CreateViewSpaceHierarchy() { if (mViewSpaceHierarchyType == BSP) { cout << "hierarchy type: Bsp" << endl; mVspBspTree = new VspBspTree(); // set view space box mVspBspTree->mBox = mViewSpaceBox; ViewCellsMap::iterator vit, vit_end = mViewCells.end(); // remove view cells and exchange them with the // view cells specialized for the current hierarchy node type for (vit = mViewCells.begin(); vit != vit_end; ++ vit) { ViewCell *vc = (*vit).second; if (!vc->IsLeaf()) // exchange only leaves continue; BspViewCell *bspVc = new BspViewCell(); bspVc->SetId(vc->GetId()); bspVc->SetPvs(vc->GetPvs()); if (vc->IsRoot()) { mViewCellsTree->mRoot = bspVc; } else { vc->GetParent()->ReplaceChildLink(vc, bspVc); } DEL_PTR(vc); (*vit).second = bspVc; } } else if (mViewSpaceHierarchyType == VSP) { cout << "hierarchy type: Vsp" << endl; mVspTree = new VspTree(); // set view space box mVspTree->mBoundingBox = mViewSpaceBox; ViewCellsMap::iterator vit, vit_end = mViewCells.end(); // reset view cells using the current node type for (vit = mViewCells.begin(); vit != vit_end; ++ vit) { ViewCell *vc = (*vit).second; VspViewCell *vspVc = new VspViewCell(); vspVc->SetPvs(vc->GetPvs()); vspVc->SetId(vc->GetId()); if (!vc->IsLeaf()) // exchange only leaves continue; if (vc->IsRoot()) { mViewCellsTree->mRoot = vspVc; } else { vc->GetParent()->ReplaceChildLink(vc, vspVc); } DEL_PTR(vc); (*vit).second = vspVc; } // if object space hierarchy already constructed if (mHierarchyManager) { mHierarchyManager->mVspTree = mVspTree; mVspTree->mHierarchyManager = mHierarchyManager; } } } void ViewCellsParseHandlers::CreateViewCellsManager() { if (mViewSpaceHierarchyType == BSP) { Debug << "creating view cells manager: VspBsp" << endl; mViewCellsManager = new VspBspViewCellsManager(mViewCellsTree, mVspBspTree); } else if (mViewSpaceHierarchyType == VSP) { Debug << "creating view cells manager: VspOsp" << endl; mViewCellsManager = new VspOspViewCellsManager(mViewCellsTree, mHierarchyManager); } mViewCellsManager->SetViewSpaceBox(mViewSpaceBox); } void ViewCellsParseHandlers::characters(const XMLCh* const chars, const unsigned int length) { mCharacterCount += length; } void ViewCellsParseHandlers::ignorableWhitespace(const XMLCh* const chars, const unsigned int length) { mSpaceCount += length; } void ViewCellsParseHandlers::resetDocument() { mAttrCount = 0; mCharacterCount = 0; mElementCount = 0; mSpaceCount = 0; } void ViewCellsParseHandlers::StartVspLeaf(AttributeList& attributes) { VspLeaf * leaf; if (mCurrentVspNode) // replace front or (if not NULL) back child { VspInterior *interior = dynamic_cast(mCurrentVspNode); leaf = new VspLeaf(interior); interior->ReplaceChildLink(NULL, leaf); } else { leaf = new VspLeaf(); mVspTree->mRoot = leaf; } ///////////// //-- find view cell associated with the id int viewCellId; int len = attributes.getLength(); for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); char *endptr = NULL; if (attrName == "viewCellId") { viewCellId = strtol(ptr, &endptr, 10); } } if (viewCellId >= 0) // valid view cell found { // TODO: get view cell with specified id /*ViewCellInterior dummyVc; dummyVc.SetId(viewCellId); ViewCellContainer::iterator vit = lower_bound(mViewCells.begin(), mViewCells.end(), &dummyVc, vlt); */ ViewCellsMap::iterator vit = mViewCells.find(viewCellId); if (vit == mViewCells.end()) cout << "error: view cell " << viewCellId << " not found" << endl; VspViewCell *viewCell = dynamic_cast((*vit).second); if (viewCell->GetId() == viewCellId) { leaf->SetViewCell(viewCell); viewCell->mLeaves.push_back(leaf); } else { cout << "error: view cell does not exist" << endl; } } else { // add to invalid view space leaf->SetViewCell(mVspTree->GetOrCreateOutOfBoundsCell()); leaf->SetTreeValid(false); mVspTree->PropagateUpValidity(leaf); } } void ViewCellsParseHandlers::StartVspInterior(AttributeList& attributes) { AxisAlignedPlane plane; int len = attributes.getLength(); for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); if (attrName == "plane") { sscanf(ptr, "%f %d", &plane.mPosition, &plane.mAxis); } } VspInterior* interior = new VspInterior(plane); if (mCurrentVspNode) // replace NULL child of parent with current node { VspInterior *parent = dynamic_cast(mCurrentVspNode); parent->ReplaceChildLink(NULL, interior); interior->SetParent(parent); AxisAlignedBox3 frontBox, backBox; parent->GetBoundingBox().Split( parent->GetPlane().mAxis, parent->GetPlane().mPosition, frontBox, backBox); if (parent->GetFront() == interior) interior->SetBoundingBox(frontBox); else interior->SetBoundingBox(backBox); } else { mVspTree->mRoot = interior; interior->SetBoundingBox(mVspTree->GetBoundingBox()); } mCurrentVspNode = interior; } void ViewCellsParseHandlers::StartOspInterior(AttributeList& attributes) { AxisAlignedPlane plane; int len = attributes.getLength(); for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); if (attrName == "plane") { sscanf(ptr, "%f %d", &plane.mPosition, &plane.mAxis); } } KdInterior* interior = new KdInterior(NULL); interior->mAxis = plane.mAxis; interior->mPosition = plane.mPosition; if (mCurrentOspNode) // replace NULL child of parent with current node { KdInterior *parent = dynamic_cast(mCurrentOspNode); parent->ReplaceChildLink(NULL, interior); interior->mParent = parent; AxisAlignedBox3 frontBox, backBox; parent->mBox.Split(parent->mAxis, parent->mPosition, frontBox, backBox); if (parent->mFront == interior) interior->mBox = frontBox; else interior->mBox = backBox; } else { mHierarchyManager->mOspTree->mRoot = interior; interior->mBox = mHierarchyManager->mOspTree->mBoundingBox; } mCurrentOspNode = interior; } void ViewCellsParseHandlers::StartOspLeaf(AttributeList& attributes) { KdLeaf * leaf = new KdLeaf(dynamic_cast(mCurrentOspNode), NULL); if (mCurrentOspNode) // replace front or (if not NULL) back child { dynamic_cast(mCurrentOspNode)->ReplaceChildLink(NULL, leaf); } else { mHierarchyManager->mOspTree->mRoot = leaf; } } void ViewCellsParseHandlers::StartBvhLeaf(AttributeList& attributes) { const int len = attributes.getLength(); Vector3 minBox, maxBox; ObjectContainer objects; for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); if (attrName == "min") { sscanf(ptr, "%f %f %f", &minBox.x, &minBox.y, &minBox.z); } if (attrName == "max") { sscanf(ptr, "%f %f %f", &maxBox.x, &maxBox.y, &maxBox.z); } if (attrName == "objects") { StartBvhLeafObjects(objects, ptr); } } AxisAlignedBox3 box = AxisAlignedBox3(minBox, maxBox); BvhLeaf *leaf; if (mCurrentBvhNode) // replace front or (if not NULL) back child { BvhInterior *interior = dynamic_cast(mCurrentBvhNode); leaf = new BvhLeaf(box, interior, (int)objects.size()); interior->ReplaceChildLink(NULL, leaf); } else { leaf = new BvhLeaf(box, NULL, (int)objects.size()); mHierarchyManager->mBvHierarchy->mRoot = leaf; } leaf->mObjects = objects; BvHierarchy::AssociateObjectsWithLeaf(leaf); } void ViewCellsParseHandlers::StartBvhLeafObjects(ObjectContainer &objects, const char *ptr) { vector objIndices; char *endptr; while (1) { const int index = strtol(ptr, &endptr, 10); if (ptr == endptr) break; objIndices.push_back(index); ptr = endptr; } //TODO: find objects and add them to pvs // TODO: get view cell with specified id MeshInstance dummyInst(NULL); vector::const_iterator it, it_end = objIndices.end(); for (it = objIndices.begin(); it != it_end; ++ it) { const int objId = *it; dummyInst.SetId(objId); ObjectContainer::iterator oit = lower_bound(mObjects->begin(), mObjects->end(), (Intersectable *)&dummyInst, ilt); if ((oit != mObjects->end()) && ((*oit)->GetId() == objId)) { objects.push_back(*oit); } else { cout << "error: object with id " << objId << " does not exist" << endl; } } } void ViewCellsParseHandlers::StartBvhInterior(AttributeList& attributes) { const int len = attributes.getLength(); Vector3 minBox, maxBox; for (int i = 0; i < len; ++ i) { string attrName(StrX(attributes.getName(i)).LocalForm()); StrX attrValue(attributes.getValue(i)); const char *ptr = attrValue.LocalForm(); if (attrName == "min") { sscanf(ptr, "%f %f %f", &minBox.x, &minBox.y, &minBox.z); } if (attrName == "max") { sscanf(ptr, "%f %f %f", &maxBox.x, &maxBox.y, &maxBox.z); } } BvhInterior* interior = new BvhInterior(AxisAlignedBox3(minBox, maxBox)); if (mCurrentBvhNode) // replace NULL child of parent with current node { BvhInterior *parent = dynamic_cast(mCurrentBvhNode); parent->ReplaceChildLink(NULL, interior); interior->SetParent(parent); } else { mHierarchyManager->mBvHierarchy->mRoot = interior; } mCurrentBvhNode = interior; } // --------------------------------------------------------------------------- // StdInParseHandlers: Overrides of the SAX ErrorHandler interface // --------------------------------------------------------------------------- void ViewCellsParseHandlers::error(const SAXParseException& e) { XERCES_STD_QUALIFIER cerr << "\nError at (file " << StrX(e.getSystemId()) << ", line " << e.getLineNumber() << ", char " << e.getColumnNumber() << "): " << StrX(e.getMessage()) << XERCES_STD_QUALIFIER endl; } void ViewCellsParseHandlers::fatalError(const SAXParseException& e) { XERCES_STD_QUALIFIER cerr << "\nFatal Error at (file " << StrX(e.getSystemId()) << ", line " << e.getLineNumber() << ", char " << e.getColumnNumber() << "): " << StrX(e.getMessage()) << XERCES_STD_QUALIFIER endl; } void ViewCellsParseHandlers::warning(const SAXParseException& e) { XERCES_STD_QUALIFIER cerr << "\nWarning at (file " << StrX(e.getSystemId()) << ", line " << e.getLineNumber() << ", char " << e.getColumnNumber() << "): " << StrX(e.getMessage()) << XERCES_STD_QUALIFIER endl; } bool ViewCellsParser::ParseViewCellsFile(const string filename, ViewCellsManager **viewCells, ObjectContainer *objects, BoundingBoxConverter *bconverter) { // Initialize the XML4C system try { XMLPlatformUtils::Initialize(); } catch (const XMLException& toCatch) { XERCES_STD_QUALIFIER cerr << "Error during initialization! Message:\n" << StrX(toCatch.getMessage()) << XERCES_STD_QUALIFIER endl; return false; } // // Create a SAX parser object. Then, according to what we were told on // the command line, set the options. // SAXParser* parser = new SAXParser; parser->setValidationScheme(valScheme); parser->setDoNamespaces(doNamespaces); parser->setDoSchema(doSchema); parser->setValidationSchemaFullChecking(schemaFullChecking); // // Create our SAX handler object and install it on the parser, as the // document and error handler. We are responsible for cleaning them // up, but since its just stack based here, there's nothing special // to do. // ViewCellsParseHandlers handler(objects, bconverter); parser->setDocumentHandler(&handler); parser->setErrorHandler(&handler); unsigned long duration; int errorCount = 0; // create a faux scope so that 'src' destructor is called before // XMLPlatformUtils::Terminate { // // Kick off the parse and catch any exceptions. Create a standard // input input source and tell the parser to parse from that. // // StdInInputSource src; try { const unsigned long startMillis = XMLPlatformUtils::getCurrentMillis(); #if USE_GZLIB XMLCh *myFilePath = XMLString::transcode(filename.c_str()); GzFileInputSource isource(myFilePath); parser->parse(isource); #else parser->parse(filename.c_str()); #endif const unsigned long endMillis = XMLPlatformUtils::getCurrentMillis(); duration = endMillis - startMillis; errorCount = parser->getErrorCount(); } catch (const OutOfMemoryException&) { XERCES_STD_QUALIFIER cerr << "OutOfMemoryException" << XERCES_STD_QUALIFIER endl; errorCount = 2; return false; } catch (const XMLException& e) { XERCES_STD_QUALIFIER cerr << "\nError during parsing: \n" << StrX(e.getMessage()) << "\n" << XERCES_STD_QUALIFIER endl; errorCount = 1; return false; } // Print out the stats that we collected and time taken if (!errorCount) { XERCES_STD_QUALIFIER cerr << filename << ": " << duration << " ms (" << handler.GetElementCount() << " elems, " << handler.GetAttrCount() << " attrs, " << handler.GetSpaceCount() << " spaces, " << handler.GetCharacterCount() << " chars)" << XERCES_STD_QUALIFIER endl; } } // // Delete the parser itself. Must be done prior to calling Terminate, below. // delete parser; XMLPlatformUtils::Terminate(); //-- assign new view cells manager *viewCells = handler.mViewCellsManager; if (errorCount > 0) return false; else return true; } }