From b793c52fe5aa20dae2aea6e49a94d27c5c718863 Mon Sep 17 00:00:00 2001 From: Amaury Johnan <amjohnen@gmail.com> Date: Tue, 3 Apr 2012 13:44:24 +0000 Subject: [PATCH] first idea of computing collapse reward too complex --- Mesh/meshGFaceRecombine.cpp | 213 ++++++++++++++++++++++++++++-------- Mesh/meshGFaceRecombine.h | 17 ++- 2 files changed, 181 insertions(+), 49 deletions(-) diff --git a/Mesh/meshGFaceRecombine.cpp b/Mesh/meshGFaceRecombine.cpp index 3a31109a0c..f3cfab8029 100644 --- a/Mesh/meshGFaceRecombine.cpp +++ b/Mesh/meshGFaceRecombine.cpp @@ -7,7 +7,7 @@ // Amaury Johnen (a.johnen@ulg.ac.be) // -#define REC2D_WAIT_TIME .01 +#define REC2D_WAIT_TIME .1 #define REC2D_NUM_ACTIO 1000 // #define REC2D_SMOOTH @@ -81,13 +81,13 @@ bool edgesInOrder(Rec2DEdge **edges, int numEdges) } return false; } - - delete v; + delete v; return true; } void crash() { + Recombine2D::drawStateOrigin(); int a[2]; int e = 0; for (int i = 0; i < 10000000; ++i) e+=a[i]; @@ -297,6 +297,7 @@ bool Recombine2D::recombine() ++_numChange; std::vector<Rec2DVertex*> newPar; nextAction->apply(newPar); + Recombine2D::incNumChange(); // forall v in newPar : check obsoletes action; @@ -335,7 +336,7 @@ double Recombine2D::recombine(int depth) FlGui::instance()->check(); time = Cpu(); #endif - Recombine2D::drawStateOrigin(); + drawStateOrigin(); Rec2DNode *root = new Rec2DNode(NULL, NULL, bestGlobalQuality, depth); Rec2DNode *currentNode = root->selectBestNode(); @@ -345,16 +346,13 @@ double Recombine2D::recombine(int depth) while (currentNode) { //_data->checkQuality(); FlGui::instance()->check(); -#if 0//def REC2D_DRAW // draw state at origin - _gf->triangles = _data->_tri; - _gf->quadrangles = _data->_quad; - CTX::instance()->mesh.changed = ENT_ALL; - drawContext::global()->draw(); +#ifdef REC2D_DRAW // draw state at origin + drawStateOrigin(); //while (Cpu()-time < REC2D_WAIT_TIME) FlGui::instance()->check(); time = Cpu(); #endif -#if 0//def REC2D_DRAW // draw all states +#ifdef REC2D_DRAW // draw all states if ( !((i+1) % ((int)std::sqrt(num)+1)) ) { dx = .0; dy -= 1.1; @@ -364,7 +362,7 @@ double Recombine2D::recombine(int depth) drawState(dx, dy); CTX::instance()->mesh.changed = ENT_ALL; drawContext::global()->draw(); - //while (Cpu()-time < REC2D_WAIT_TIME) + while (Cpu()-time < REC2D_WAIT_TIME) FlGui::instance()->check(); ++i; time = Cpu(); @@ -783,17 +781,24 @@ void Rec2DData::checkEntities() void Rec2DData::printActions() const { + Msg::Info("size %d", _actions.size()); std::map<int, std::vector<double> > data; std::list<Rec2DAction*>::const_iterator it = _actions.begin(); for (; it != _actions.end(); ++it) { + Msg::Info("action %d", *it); + } + it = _actions.begin(); + for (; it != _actions.end(); ++it) { + Msg::Info("action %d", *it); std::vector<Rec2DElement*> tri; (*it)->getElements(tri); - Msg::Info("action %d (%d, %d) -> reward %g", *it, tri[0]->getNum(), tri[1]->getNum(), .0/*(*it)->getReward()*/); + Msg::Info("action %d (%d, %d)", *it, tri[0]->getNum(), tri[1]->getNum()); + Msg::Info("action %d (%d, %d) -> reward %g", *it, tri[0]->getNum(), tri[1]->getNum(), (*it)->getReward()); //Msg::Info("action %d -> reward %g", *it, (*it)->getReward()); data[tri[0]->getNum()].resize(1); data[tri[1]->getNum()].resize(1); - data[tri[0]->getNum()][0] = (*it)->getReward(); - data[tri[1]->getNum()][0] = (*it)->getReward(); + //data[tri[0]->getNum()][0] = (*it)->getReward(); + //data[tri[1]->getNum()][0] = (*it)->getReward(); //(*it)->print(); } new PView("Jmin_bad", "ElementData", Recombine2D::getGFace()->model(), data); @@ -1527,6 +1532,9 @@ Rec2DAction::Rec2DAction() bool Rec2DAction::operator<(Rec2DAction &other) { + if (!&other) Msg::Info("not other"); + if (!this) Msg::Info("not this"); + Rec2DData::printAction(); return getReward() < other.getReward(); } @@ -1819,11 +1827,17 @@ void Rec2DTwoTri2Quad::apply(Rec2DDataChange *rdc, std::vector<Rec2DAction*> *&createdActions) const { if (isObsolete()) { + printIdentity(); + int p[4]; + p[0] = _vertices[0]->getParity(); + p[1] = _vertices[1]->getParity(); + p[2] = _vertices[2]->getParity(); + p[3] = _vertices[3]->getParity(); + Msg::Info("%d %d %d %d", p[0], p[1], p[2], p[3]); Msg::Error("[Rec2DTwoTri2Quad] I was obsolete !"); + crash(); } - else { - _doWhatYouHaveToDoWithParity(rdc); - } + _doWhatYouHaveToDoWithParity(rdc); #ifdef REC2D_DRAW rdc->setAction(this); #endif @@ -1835,7 +1849,7 @@ void Rec2DTwoTri2Quad::apply(Rec2DDataChange *rdc, rdc->hide(_triangles[1]); rdc->hide(_edges[4]); rdc->append(new Rec2DElement((MQuadrangle*)NULL, (const Rec2DEdge**)_edges)); - Recombine2D::incNumChange(); + rdc->checkObsoleteActions(_vertices, 4); } void Rec2DTwoTri2Quad::_doWhatYouHaveToDoWithParity(Rec2DDataChange *rdc) const @@ -1867,9 +1881,6 @@ void Rec2DTwoTri2Quad::_doWhatYouHaveToDoWithParity(Rec2DDataChange *rdc) const Rec2DData::associateParity(_vertices[i+j]->getParity(), par, rdc); } } - rdc->checkObsoleteActions(_vertices, 4); - static int a = -1; - if (++a < 1) Msg::Warning("FIXME should check obsolete ?"); } } @@ -2079,13 +2090,63 @@ void Rec2DCollapse::_computeGlobQual() std::vector<Rec2DVertex*> verts; _rec->_vertices[0]->getMoreNeighbourVertices(verts); _rec->_vertices[1]->getMoreNeighbourVertices(verts); - for (unsigned int i = 0; i < verts.size(); ++i) { - if (verts[i]->getLastUpdate() > _lastUpdate) { - delete (new Rec2DNode(NULL, this, _globQualIfExecuted, 0)); - _lastUpdate = Recombine2D::getNumChange(); - return; + unsigned int i = 0; + while (i < verts.size() && verts[i]->getLastUpdate() <= _lastUpdate) ++i; + if (i >= verts.size()) + return; + + /*int b0 = _rec->_vertices[0]->getOnBoundary(); + int b1 = _rec->_vertices[1]->getOnBoundary();*/ + std::vector<Rec2DEdge*> edges; + _rec->_vertices[0]->getUniqueEdges(edges); + _rec->_vertices[1]->getUniqueEdges(edges); + + int numEdge = 0; + double valEdge = .0, valVert = .0; + for (i = 0; i < edges.size(); ++i) { + valEdge -= edges[i]->getWeightedQual(); + numEdge -= edges[i]->getWeight(); + } + valVert -= _rec->_vertices[0]->getQual(); + valVert -= _rec->_vertices[1]->getQual(); + //valVert += _vertices[2]->getGain(-1); + //valVert += _vertices[3]->getGain(-1); + + /*std::vector<Rec2DVertex*> neighbourVert; + _vertices[0]->getMoreNeighbourVertices(neighbourVert); + _vertices[1]->getMoreNeighbourVertices(neighbourVert); + unsigned int i = 0; + while (i < neighbourVert.size()) { + if (neighbourVert[i] == _vertices[0] || + neighbourVert[i] == _vertices[1] || + neighbourVert[i] == _vertices[2] || + neighbourVert[i] == _vertices[3] ) { + neighbourVert[i] = neighbourVert.back(); + neighbourVert.pop_back(); + } + else + ++i; + }*/ + + std::vector<std::pair<Rec2DVertex*, int> > vert2weight; + _rec->_vertices[0]->getNeighbourWeight(vert2weight); + _rec->_vertices[1]->getNeighbourWeight(vert2weight); + i = 0; + while (i < vert2weight.size()) { + if (vert2weight[i].first == _rec->_vertices[0] || + vert2weight[i].first == _rec->_vertices[1] ) { + vert2weight[i] = vert2weight.back(); + vert2weight.pop_back(); } + else + ++i; } + _computeQualEdges(numEdge, valEdge, vert2weight, newVert); + _computeQualVertices(valVert, vert2weight, point); + + _globQualIfExecuted = Rec2DData::getGlobalQuality(numEdge, valEdge, -1, valVert); + + _lastUpdate = Recombine2D::getNumChange(); } void Rec2DCollapse::printIdentity() const @@ -2103,7 +2164,19 @@ void Rec2DCollapse::apply(Rec2DDataChange *rdc, std::vector<Rec2DAction*> *&createdActions) const { if (isObsolete()) { + printIdentity(); + int p[2]; + p[0] = _rec->_vertices[0]->getParity(); + p[1] = _rec->_vertices[1]->getParity(); + Msg::Info("%d %d %d %d", _rec->_vertices[0]->getNum(), _rec->_vertices[1]->getNum(), _rec->_vertices[2]->getNum(), _rec->_vertices[3]->getNum()); + Msg::Info("%d %d - %d %d %d", p[0], p[1], + _rec->_vertices[0]->getOnBoundary() && _rec->_vertices[1]->getOnBoundary(), + !_rec->_vertices[2]->getOnBoundary() && _rec->_vertices[2]->getNumElements() < 4, + !_rec->_vertices[3]->getOnBoundary() && _rec->_vertices[3]->getNumElements() < 4); + _rec->_vertices[2]->printElem(); + _rec->_vertices[3]->printElem(); Msg::Error("[Rec2DTwoTri2Quad] I was obsolete !"); + crash(); } #ifdef REC2D_DRAW rdc->setAction(this); @@ -2188,17 +2261,13 @@ void Rec2DCollapse::apply(Rec2DDataChange *rdc, int parKO, parOK; if ((parKO = vKO->getParity())) { - if (!(parOK = vOK->getParity())) { + if (!(parOK = vOK->getParity())) rdc->changeParity(vOK, parKO); - Rec2DVertex *v[1] = {vOK}; - rdc->checkObsoleteActions(v, 1); - } - else if (parOK/2 != parKO/2) { + else if (parOK/2 != parKO/2) Rec2DData::associateParity(std::max(parOK, parKO), std::min(parOK, parKO), rdc); - } } - Recombine2D::incNumChange(); + rdc->checkObsoleteActions(_rec->_vertices, 4); } bool Rec2DCollapse::isObsolete() const @@ -2221,6 +2290,30 @@ bool Rec2DCollapse::_hasIdenticalElement() const _rec->_triangles[1]->hasIdenticalNeighbour() ; } +void Rec2DCollapse::_computeQualEdges( + int &numEdge, double &valEdge, + std::vector<std::pair<Rec2DVertex*, int> > &v2weight, + Rec2DVertex *rv ) +{ + for (unsigned int i = 0; i < v2weight.size(); ++i) { + numEdge += v2weight[i].second; + Rec2DEdge re = Rec2DEdge(rv, v2weight[i].first); + valEdge += re.getQual(); + } +} + +void Rec2DCollapse::_computeQualVertices( + double &valVert, + std::vector<std::pair<Rec2DVertex*, int> > &vert, + SPoint2 p ) +{ + for (unsigned int i = 0; i < vert.size(); ++i) { + valVert += vert[i].first->getChangeQual(p, _rec->_vertices[0], + _rec->_vertices[1] ); + } + valVert; +} + bool Rec2DCollapse::whatWouldYouDo (std::map<Rec2DVertex*, std::vector<int> > &suggestions) { @@ -2719,6 +2812,22 @@ void Rec2DVertex::getMoreNeighbourVertices(std::vector<Rec2DVertex*> &verts) con } } +void Rec2DVertex::getNeighbourWeight( + std::vector<std::pair<Rec2DVertex*, int> > &vect) const +{ + unsigned int size = vect.size(); + for (unsigned int i = 0; i < _edges.size(); ++i) { + Rec2DVertex *rv = _edges[i]->getOtherVertex(this); + unsigned int index = 0; + while (index < size && vect[index].first != rv) ++index; + + if (index >= size) + vect.push_back(std::make_pair(rv, _edges[i]->getWeight())); + else + vect[index].second += _edges[i]->getWeight() - REC2D_EDGE_BASE; + } +} + void Rec2DVertex::getUniqueEdges(std::vector<Rec2DEdge*> &edges) const { unsigned int size = edges.size(); @@ -2870,6 +2979,13 @@ void Rec2DVertex::rmv(const Rec2DElement *rel) Msg::Error("[Rec2DVertex] Didn't removed element, didn't have it"); } +void Rec2DVertex::printElem() const +{ + for (unsigned int i = 0; i < _elements.size(); ++i) { + Msg::Info("%d - %d", i, _elements[i]->getNum()); + } +} + void Rec2DVertex::getUniqueActions(std::vector<Rec2DAction*> &actions) const { std::vector<Rec2DAction*> actions2; @@ -3249,19 +3365,19 @@ void Rec2DElement::getVertices(std::vector<Rec2DVertex*> &verts) const int i = 0, k = 0; while (k < _numEdge) { verts[k] = _edges[i/2]->getVertex(i%2); - bool b = true; + bool itsOK = true; for (int j = 0; j < k; ++j) { if (verts[k] == verts[j]) - b = false; + itsOK = false; } - if (b) { - ++k; + if (itsOK) { // make sure they are well ordered (edges are ordered) if (k == 2 && _edges[i/2]->getVertex((i+1)%2) == verts[0]) { Rec2DVertex *rv = verts[0]; verts[0] = verts[1]; verts[1] = rv; } + ++k; } ++i; } @@ -3431,6 +3547,7 @@ Rec2DNode::Rec2DNode(Rec2DNode *father, Rec2DAction *ra, _globalQuality = Rec2DData::getGlobalQuality(); if (depth) { + Recombine2D::incNumChange(); std::vector<Rec2DAction*> actions; Recombine2D::nextTreeActions(actions, neighbours); @@ -3461,9 +3578,10 @@ Rec2DNode::Rec2DNode(Rec2DNode *father, Rec2DAction *ra, bestEndGlobQual = _bestEndGlobQual; if (_dataChange) { - if (!Rec2DData::revertDataChange(_dataChange)) { - Msg::Error(" 1 - don't reverted changes");Msg::Error(" "); - } + if (depth) + Recombine2D::incNumChange(); + if (!Rec2DData::revertDataChange(_dataChange)) + Msg::Error(" 1 - don't reverted changes"); _dataChange = NULL; } } @@ -3537,6 +3655,7 @@ void Rec2DNode::develop(int depth, double &bestEndGlobQual) } } else { + Recombine2D::incNumChange(); std::vector<Rec2DAction*> actions; Recombine2D::nextTreeActions(actions, neighbours); @@ -3562,9 +3681,9 @@ void Rec2DNode::develop(int depth, double &bestEndGlobQual) bestEndGlobQual = _bestEndGlobQual; if (delChanges) { - if (!Rec2DData::revertDataChange(_dataChange)) { + Recombine2D::incNumChange(); + if (!Rec2DData::revertDataChange(_dataChange)) Msg::Error(" 1 - don't reverted changes");Msg::Error(" "); - } _dataChange = NULL; } } @@ -3579,11 +3698,12 @@ bool Rec2DNode::makeChanges() _ra->color(0, 0, 200); CTX::instance()->mesh.changed = ENT_ALL; drawContext::global()->draw(); - while (Cpu()-time < REC2D_WAIT_TIME) + //while (Cpu()-time < REC2D_WAIT_TIME) FlGui::instance()->check(); #endif _ra->apply(_dataChange, _createdActions); Rec2DData::setNumTri(_remainingTri); + Recombine2D::incNumChange(); return true; } @@ -3591,9 +3711,10 @@ bool Rec2DNode::revertChanges() { if (!_dataChange) return false; - if (!Rec2DData::revertDataChange(_dataChange)) { - Msg::Error(" 1 - don't reverted changes");Msg::Error(" "); - } + if (!Rec2DData::revertDataChange(_dataChange)) + Msg::Error(" 1 - don't reverted changes"); + else + Recombine2D::incNumChange(); _dataChange = NULL; return true; } diff --git a/Mesh/meshGFaceRecombine.h b/Mesh/meshGFaceRecombine.h index b90ba829e9..1b1562792b 100644 --- a/Mesh/meshGFaceRecombine.h +++ b/Mesh/meshGFaceRecombine.h @@ -114,6 +114,7 @@ class Rec2DData { void printState() const; void printActions() const; + static void printAction() {_current->printActions();} void sortActions() {_actions.sort(lessRec2DAction());} void drawTriangles(double shiftx, double shifty) const; void drawElements(double shiftx, double shifty) const; @@ -299,7 +300,7 @@ class Rec2DAction { public : Rec2DAction(); - virtual ~Rec2DAction() {} + virtual ~Rec2DAction() {Msg::Error("deleting %d", this);} virtual void hide() = 0; virtual void reveal() = 0; virtual bool checkCoherence() const = 0; @@ -345,7 +346,7 @@ class Rec2DTwoTri2Quad : public Rec2DAction { public : Rec2DTwoTri2Quad(Rec2DElement*, Rec2DElement*); - ~Rec2DTwoTri2Quad() {if (_col) Msg::Error("[Rec2DTwoTri2Quad] have collapse");hide();} + ~Rec2DTwoTri2Quad() {Msg::Error("deleting %d", this);if (_col) Msg::Error("[Rec2DTwoTri2Quad] have collapse");hide();} virtual void hide(); virtual void reveal(); virtual bool checkCoherence() const; @@ -386,7 +387,7 @@ class Rec2DCollapse : public Rec2DAction { public : Rec2DCollapse(Rec2DTwoTri2Quad*); - ~Rec2DCollapse() {_rec->_col = NULL; hide();} + ~Rec2DCollapse() {Msg::Error("deleting %d", this);_rec->_col = NULL; hide();} virtual void hide(); virtual void reveal(); virtual bool checkCoherence() const {return _rec->checkCoherence();} @@ -429,6 +430,12 @@ class Rec2DCollapse : public Rec2DAction { private : virtual void _computeGlobQual(); bool _hasIdenticalElement() const; + void _computeQualEdges(int &numEdge, double &valEdge, + std::vector<std::pair<Rec2DVertex*, int> > &v2weight, + SPoint2 ); + void _computeQualVertices(double &valVert, + std::vector<std::pair<Rec2DVertex*, int> > &v2weight, + SPoint2 ); }; class Rec2DEdge { @@ -526,6 +533,7 @@ class Rec2DVertex { void getTriangles(std::set<Rec2DElement*>&) const; void getElements(std::vector<Rec2DElement*>&) const; inline MVertex* getMVertex() const {return _v;} + void getNeighbourWeight(std::vector<std::pair<Rec2DVertex*, int> >&) const; inline int getLastUpdate() const {return _lastUpdate;} inline void getxyz(double *xyz) const { @@ -536,6 +544,7 @@ class Rec2DVertex { inline double u() const {return _param[0];} inline double v() const {return _param[1];} void relocate(SPoint2 p); + inline SPoint2 getParam() {return _param;} inline void getParam(SPoint2 *p) {*p = _param;} void add(const Rec2DEdge*); @@ -546,6 +555,8 @@ class Rec2DVertex { bool has(const Rec2DElement*) const; void rmv(const Rec2DElement*); + void printElem() const; + void getUniqueActions(std::vector<Rec2DAction*>&) const; static void initStaticTable(); -- GitLab