yade-dev team mailing list archive
  
  - 
     yade-dev team yade-dev team
- 
    Mailing list archive
  
- 
    Message #10134
  
 [Branch ~yade-pkg/yade/git-trunk] Rev 3714: Recover	DomainLimiter	(LawTester should be moved somewhere).
  
------------------------------------------------------------
revno: 3714
committer: Anton Gladky <gladky.anton@xxxxxxxxx>
timestamp: Tue 2013-10-15 07:01:48 +0200
message:
  Recover DomainLimiter (LawTester should be moved somewhere).
added:
  pkg/dem/DomainLimiter.cpp
  pkg/dem/DomainLimiter.hpp
--
lp:yade
https://code.launchpad.net/~yade-pkg/yade/git-trunk
Your team Yade developers is subscribed to branch lp:yade.
To unsubscribe from this branch go to https://code.launchpad.net/~yade-pkg/yade/git-trunk/+edit-subscription
=== added file 'pkg/dem/DomainLimiter.cpp'
--- pkg/dem/DomainLimiter.cpp	1970-01-01 00:00:00 +0000
+++ pkg/dem/DomainLimiter.cpp	2013-10-15 05:01:48 +0000
@@ -0,0 +1,343 @@
+#include<yade/pkg/dem/DomainLimiter.hpp>
+#include<yade/pkg/dem/Shop.hpp>
+
+YADE_PLUGIN((DomainLimiter)(LawTester)
+	#ifdef YADE_OPENGL
+		(GlExtra_LawTester)(GlExtra_OctreeCubes)
+	#endif
+);
+
+void DomainLimiter::action(){
+	std::list<Body::id_t> out;
+	FOREACH(const shared_ptr<Body>& b, *scene->bodies){
+		if((!b) or ((mask>0) and ((b->groupMask & mask)==0))) continue;
+		const Sphere* sphere = dynamic_cast<Sphere*>(b->shape.get()); 
+		if (sphere){ //Delete only spheres
+			const Vector3r& p(b->state->pos);
+			if(p[0]<lo[0] || p[0]>hi[0] || p[1]<lo[1] || p[1]>hi[1] || p[2]<lo[2] || p[2]>hi[2]) {
+				out.push_back(b->id);
+				nDeleted++;
+				mDeleted+=b->state->mass;
+				Real r = sphere->radius; vDeleted+=(4/3.)*Mathr::PI*pow(r,3);
+			}
+		}
+	}
+	FOREACH(Body::id_t id, out){
+		scene->bodies->erase(id);
+	}
+}
+
+#include<yade/pkg/dem/DemXDofGeom.hpp>
+#include<yade/pkg/dem/ScGeom.hpp>
+#include<yade/pkg/dem/L3Geom.hpp>
+#include<yade/pkg/common/NormShearPhys.hpp>
+#include<yade/lib/smoothing/LinearInterpolate.hpp>
+#include<yade/lib/pyutil/gil.hpp>
+
+CREATE_LOGGER(LawTester);
+
+void LawTester::postLoad(LawTester&){
+	if(ids.size()==0) return; // uninitialized object, don't do nothing at all
+	if(ids.size()!=2) throw std::invalid_argument("LawTester.ids: exactly two values must be given.");
+	if(disPath.empty() && rotPath.empty()) throw invalid_argument("LawTester.{disPath,rotPath}: at least one point must be given.");
+	if(pathSteps.empty()) throw invalid_argument("LawTester.pathSteps: at least one value must be given.");
+	size_t pathSize=max(disPath.size(),rotPath.size());
+	// update path points
+	_path.clear(); _path.push_back(Vector6r::Zero());
+	for(size_t i=0; i<pathSize; i++) {
+		Vector6r pt;
+		pt.head<3>()=Vector3r(i<disPath.size()?disPath[i]:(disPath.empty()?Vector3r::Zero():*(disPath.rbegin())));
+		pt.tail<3>()=Vector3r(i<rotPath.size()?rotPath[i]:(rotPath.empty()?Vector3r::Zero():*(rotPath.rbegin())));
+		_path.push_back(pt);
+
+	}
+	// update time points from distances, repeat last distance if shorter than path
+	_pathT.clear(); _pathT.push_back(0);
+	for(size_t i=0; i<pathSteps.size(); i++) _pathT.push_back(_pathT[i]+pathSteps[i]);
+	int lastDist=pathSteps[pathSteps.size()-1];
+	for(size_t i=pathSteps.size(); i<pathSize; i++) _pathT.push_back(*(_pathT.rbegin())+lastDist);
+}
+
+void LawTester::action(){
+	Vector2r foo; // avoid undefined ~Vector2r with clang?
+	if(ids.size()!=2) throw std::invalid_argument("LawTester.ids: exactly two values must be given.");
+	LOG_DEBUG("=================== LawTester step "<<step<<" ========================");
+	const shared_ptr<Interaction> Inew=scene->interactions->find(ids[0],ids[1]);
+	string strIds("##"+lexical_cast<string>(ids[0])+"+"+lexical_cast<string>(ids[1]));
+	// interaction not found at initialization
+	if(!I && (!Inew || !Inew->isReal())){
+		LOG_WARN("Interaction "<<strIds<<" does not exist (yet?), no-op."); return;
+		//throw std::runtime_error("LawTester: interaction "+strIds+" does not exist"+(Inew?" (to be honest, it does exist, but it is not real).":"."));
+	}
+	// interaction was deleted meanwhile
+	if(I && (!Inew || !Inew->isReal())) throw std::runtime_error("LawTester: interaction "+strIds+" was deleted"+(Inew?" (is not real anymore).":"."));
+	// different interaction object
+	if(I && Inew && I!=Inew) throw std::logic_error("LawTester: interacion "+strIds+" is a different object now?!");
+	assert(Inew);
+	bool doInit=(!I);
+	if(doInit) I=Inew;
+
+	id1=I->getId1(); id2=I->getId2();
+	// test object types
+	GenericSpheresContact* gsc=dynamic_cast<GenericSpheresContact*>(I->geom.get());
+	ScGeom* scGeom=dynamic_cast<ScGeom*>(I->geom.get());
+	L3Geom* l3Geom=dynamic_cast<L3Geom*>(I->geom.get());
+	L6Geom* l6Geom=dynamic_cast<L6Geom*>(I->geom.get());
+	ScGeom6D* scGeom6d=dynamic_cast<ScGeom6D*>(I->geom.get());
+	bool hasRot=(l6Geom || scGeom6d);
+	//NormShearPhys* phys=dynamic_cast<NormShearPhys*>(I->phys.get());			//Disabled because of warning
+	if(!gsc) throw std::invalid_argument("LawTester: IGeom of "+strIds+" not a GenericSpheresContact.");
+	if(!scGeom && !l3Geom) throw std::invalid_argument("LawTester: IGeom of "+strIds+" is neither ScGeom, nor Dem3DofGeom, nor L3Geom (or L6Geom).");
+	assert(!((bool)scGeom && (bool)l3Geom)); // nonsense
+	// get body objects
+	State *state1=Body::byId(id1,scene)->state.get(), *state2=Body::byId(id2,scene)->state.get();
+	scene->forces.sync();
+	if(state1->blockedDOFs!=State::DOF_ALL) { LOG_INFO("Blocking all DOFs for #"<<id1); state1->blockedDOFs=State::DOF_ALL;}
+	if(state2->blockedDOFs!=State::DOF_ALL) { LOG_INFO("Blocking all DOFs for #"<<id2); state2->blockedDOFs=State::DOF_ALL;}
+
+
+	if(step-1>*(_pathT.rbegin())){
+		LOG_INFO("Last step done, setting zero velocities on #"<<id1<<", #"<<id2<<".");
+		state1->vel=state1->angVel=state2->vel=state2->angVel=Vector3r::Zero();
+		uTest=uTestNext;
+		if(doneHook.empty()){ LOG_INFO("No doneHook set, dying."); dead=true; }
+		else{ LOG_INFO("Running doneHook: "<<doneHook);	pyRunString(doneHook);}
+		return;
+	}
+	/* initialize or update local axes and trsf */
+	uGeom.tail<3>()=Vector3r(NaN,NaN,NaN);
+	if(!l3Geom){ // IGeom's that don't have local axes
+		axX=gsc->normal; /* just in case */ axX.normalize();
+		if(doInit){ // initialization of the new interaction -- define local axes
+			// take vector in the y or z direction, depending on its length; arbitrary, but one of them is sure to be non-zero
+			axY=axX.cross(abs(axX[1])<abs(axX[2])?Vector3r::UnitY():Vector3r::UnitZ());
+			axY.normalize();
+			axZ=axX.cross(axY);
+			LOG_DEBUG("Initial axes x="<<axX<<", y="<<axY<<", z="<<axZ);
+			if(scGeom6d) uGeom.tail<3>()=Vector3r::Zero();
+		} else { // udpate of an existing interaction
+			if(scGeom){
+				scGeom->rotate(axY); scGeom->rotate(axZ);
+				scGeom->rotate(shearTot);
+				shearTot+=scGeom->shearIncrement();
+				uGeom.head<3>()=Vector3r(-scGeom->penetrationDepth,shearTot.dot(axY),shearTot.dot(axZ));
+				if(scGeom6d) uGeom.tail<3>()=-1.*Vector3r(scGeom6d->getTwist(),scGeom6d->getBending().dot(axY),scGeom6d->getBending().dot(axZ));
+			}
+			else{ // d3dGeom
+				throw runtime_error("Geom type not yet supported.");
+			}
+		}
+		// update the transformation
+		// the matrix is orthonormal, since axX, axY are normalized and and axZ is their cross-product
+		trsf.row(0)=axX; trsf.row(1)=axY; trsf.row(2)=axZ;
+	} else {
+		trsf=Matrix3r(l3Geom->trsf);
+		axX=trsf.row(0); axY=trsf.row(1); axZ=trsf.row(2);
+		uGeom.head<3>()=l3Geom->u;
+		if(l6Geom) uGeom.tail<3>()=l6Geom->phi;
+	}
+	// perform all shearing by translation, as it does not induce bending
+	if(hasRot && rotWeight!=0){ LOG_INFO("LawTester.rotWeight set to 0 (was "<<rotWeight<<"), since rotational DoFs are in use."); rotWeight=0; }
+	contPt=gsc->contactPoint;
+	refLength=gsc->refR1+gsc->refR2;
+	renderLength=.5*refLength;
+	
+	// here we go ahead, finally
+	Vector6r uu=linearInterpolate<Vector6r,int>(step,_pathT,_path,_interpPos);
+	Vector6r dUU=uu-uuPrev; uuPrev=uu;
+	Vector3r dU(dUU.head<3>()), dPhi(dUU.tail<3>());
+	//Vector3r dU=u-uPrev.head<3>(); uPrev.head<3>()=u;
+	//Vector3r dPhi=phi-uPrev.tail<3>(); uPrev.tail<3>()=phi;
+	if(displIsRel){
+		LOG_DEBUG("Relative displacement diff is "<<dU<<" (will be normalized by "<<gsc->refR1+gsc->refR2<<")");
+		dU*=refLength;
+	}
+	LOG_DEBUG("Absolute diff is: displacement "<<dU<<", rotation "<<dPhi);
+	uTest=uTestNext; // the value that was next in the previous step is the current one now
+	uTestNext.head<3>()+=dU; uTestNext.tail<3>()+=dPhi;
+
+	// reset velocities where displacement is controlled
+	//for(int i=0; i<3; i++){ if(forceControl[i]==0){ state1.vel[i]=0; state2.vel[i]=0; }
+	
+	// shear is applied as rotation of id2: dε=r₁dθ → dθ=dε/r₁;
+	Vector3r vel[2],angVel[2];
+	//State* states[]={state1,state2};
+	for(int i=0; i<2; i++){
+		int sign=(i==0?-1:1);
+		Real weight=(i==0?1-idWeight:idWeight);
+		// FIXME: this should not use refR1, but real CP-particle distance perhaps?
+		Real radius=(i==0?gsc->refR1:gsc->refR2);
+		Real relRad=radius/refLength;
+		// signed and weighted displacement/rotation to be applied on this sphere (reversed for #0)
+		// some rotations must cancel the sign, by multiplying by sign again
+		Vector3r ddU=sign*dU*weight;
+
+		// twist can be still distributed with idWeight (!)
+		Vector3r ddPhi=sign*dPhi*(1-relRad); /* shear angles must distribute to both, otherwise it would induce shear */
+		ddPhi[0]=sign*dPhi[0]*weight; // twist can be still distributed with idWeight
+		vel[i]=angVel[i]=Vector3r::Zero();
+
+		// normal displacement
+
+		vel[i]+=axX*ddU[0]/scene->dt;
+
+		// shear rotation
+
+		//   multiplication by sign cancels sign in ddU, since rotation is non-symmetric (to increase shear, both spheres have the same rotation)
+		//   (unlike shear displacement, which is symmetric)
+		// rotation around Z (which gives y-shear) must be inverted: +ry gives +εzm while -rz gives +εy
+		Real rotZ=-sign*rotWeight*ddU[1]/radius, rotY=sign*rotWeight*ddU[2]/radius;
+		angVel[i]+=(rotY*axY+rotZ*axZ)/scene->dt;
+
+		// shear displacement
+
+		// angle that is traversed by a sphere in order to give desired ddU when displaced on the branch of r1+r2
+		// FIXME: is the branch value correct here?!
+		Real arcAngleY=atan((1-rotWeight)*ddU[1]/radius), arcAngleZ=atan((1-rotWeight)*ddU[2]/radius);
+		// same, but without the atan, which can be disregarded for small increments:
+		//    Real arcAngleY=(1-rotWeight)*ddU[1]/radius, arcAngleZ=(1-rotWeight)*ddU[2]/radius; 
+		vel[i]+=axY*radius*sin(arcAngleY)/scene->dt; vel[i]+=axZ*radius*sin(arcAngleZ)/scene->dt;
+
+		// compensate distance increase caused by motion along the perpendicular axis
+		// cos(argAngle*) is always positive, regardless of the orientation
+		// and the compensation is always in the -εx sense (-sign → +1 for #0, -1 for #1)
+		vel[i]+=-sign*axX*radius*((1-cos(arcAngleY))+(1-cos(arcAngleZ)))/scene->dt;
+
+		// rotation, convert from local to global
+		angVel[i]+=trsf.transpose()*ddPhi;
+
+		LOG_DEBUG("vel="<<vel[i]<<", angVel="<<angVel[i]<<", rotY,rotZ="<<rotY<<","<<rotZ<<", arcAngle="<<arcAngleY<<","<<arcAngleZ<<", sign="<<sign<<", weight="<<weight);
+
+	}
+	state1->vel=vel[0]; state1->angVel=angVel[0];
+	state2->vel=vel[1]; state2->angVel=angVel[1];
+	LOG_DEBUG("Body #"<<id1<<", setting vel="<<vel[0]<<", angVel="<<angVel[0]);
+	LOG_DEBUG("Body #"<<id2<<", setting vel="<<vel[1]<<", angVel="<<angVel[1]);
+
+	/* find out where are we at in the path, run hooks if approriate */
+	// _pathT has the first (zero) value added by us, so we skip it
+	int nPathT=_pathT.size();
+	for(int i=1; i<nPathT; i++){
+		// i-th point on _pathT is (i-1)th on path; run corresponding hook, if it exists
+		if(step==_pathT[i] && ((int) hooks.size())>(i-1) && !hooks[i-1].empty()) pyRunString(hooks[i-1]);
+	}
+	step++;
+}
+
+#ifdef YADE_OPENGL
+#include<yade/lib/opengl/OpenGLWrapper.hpp>
+#include<yade/lib/opengl/GLUtils.hpp>
+#include<yade/pkg/common/GLDrawFunctors.hpp>
+#include<yade/pkg/common/OpenGLRenderer.hpp>
+#include<GL/glu.h>
+
+CREATE_LOGGER(GlExtra_LawTester);
+
+void GlExtra_LawTester::render(){
+	// scene object changed (after reload, for instance), for re-initialization
+	if(tester && tester->scene!=scene) tester=shared_ptr<LawTester>();
+
+	if(!tester){ FOREACH(shared_ptr<Engine> e, scene->engines){ tester=dynamic_pointer_cast<LawTester>(e); if(tester) break; } }
+	if(!tester){ LOG_ERROR("No LawTester in O.engines, killing myself."); dead=true; return; }
+
+	//if(tester->renderLength<=0) return;
+	glColor3v(Vector3r(1,0,1));
+
+	// switch to local coordinates
+	glTranslatev(tester->contPt);
+	//glMultMatrixd(Eigen::Affine3d(tester->trsf.transpose()).data());
+	glMultMatrix(Eigen::Transform<Real,3,Eigen::Affine>(tester->trsf.transpose()).data());
+
+
+	glDisable(GL_LIGHTING); 
+	//glColor3v(Vector3r(1,0,1));
+	//glBegin(GL_LINES); glVertex3v(Vector3r::Zero()); glVertex3v(.1*Vector3r::Ones()); glEnd(); 
+	//GLUtils::GLDrawText(string("This is the contact point!"),Vector3r::Zero(),Vector3r(1,0,1));
+
+	// local axes
+	glLineWidth(2.);
+	for(int i=0; i<3; i++){
+		Vector3r pt=Vector3r::Zero(); pt[i]=.5*tester->renderLength; Vector3r color=.3*Vector3r::Ones(); color[i]=1;
+		GLUtils::GLDrawLine(Vector3r::Zero(),pt,color);
+		GLUtils::GLDrawText(string(i==0?"x":(i==1?"y":"z")),pt,color);
+	}
+
+	// put the origin to the initial (no-shear) point, so that the current point appears at the contact point
+	glTranslatev(Vector3r(0,tester->uTestNext[1],tester->uTestNext[2]));
+
+
+	const int t(tester->step); const vector<int>& TT(tester->_pathT); const vector<Vector6r>& VV(tester->_path);
+	size_t numSegments=TT.size();
+	const Vector3r colorBefore=Vector3r(.7,1,.7), colorAfter=Vector3r(1,.7,.7);
+
+	// scale displacement, if they have the strain meaning
+	Real scale=1;
+	if(tester->displIsRel) scale=tester->refLength;
+
+	// find maximum displacement, draw axes in the shear plane
+	Real displMax=0;
+	FOREACH(const Vector6r& v, VV) displMax=max(v.head<3>().squaredNorm(),displMax);
+	displMax=1.2*scale*sqrt(displMax);
+
+	glLineWidth(1.);
+	GLUtils::GLDrawLine(Vector3r(0,-displMax,0),Vector3r(0,displMax,0),Vector3r(.5,0,0));
+	GLUtils::GLDrawLine(Vector3r(0,0,-displMax),Vector3r(0,0,displMax),Vector3r(.5,0,0));
+
+	// draw displacement path
+	glLineWidth(4.);
+	for(size_t segment=0; segment<numSegments-1; segment++){
+		// different colors before and after the current point
+		Real t0=TT[segment],t1=TT[segment+1];
+		const Vector3r &from=-VV[segment].head<3>()*scale, &to=-VV[segment+1].head<3>()*scale;
+		// current segment
+		if(t>t0 && t<t1){
+			Real norm=(t-t0)/(t1-t0);
+			GLUtils::GLDrawLine(from,from+(to-from)*norm,colorBefore);
+			GLUtils::GLDrawLine(from+(to-from)*norm,to,colorAfter);
+		} else {	// other segment
+			GLUtils::GLDrawLine(from,to,t<t0?colorAfter:colorBefore);
+		}
+	}
+
+	glLineWidth(1.);
+}
+
+
+void GlExtra_OctreeCubes::postLoad(GlExtra_OctreeCubes&){
+	if(boxesFile.empty()) return;
+	boxes.clear();
+	ifstream txt(boxesFile.c_str());
+	while(!txt.eof()){
+		Real data[8];
+		for(int i=0; i<8; i++){ if(i<7 && txt.eof()) goto done; txt>>data[i]; }
+		OctreeBox ob; Vector3r mn(data[0],data[1],data[2]), mx(data[3],data[4],data[5]);
+		ob.center=.5*(mn+mx); ob.extents=(.5*(mx-mn)); ob.level=(int)data[6]; ob.fill=(int)data[7];
+		// for(int i=0; i<=ob.level; i++) cerr<<"\t"; cerr<<ob.level<<": "<<mn<<"; "<<mx<<"; "<<ob.center<<"; "<<ob.extents<<"; "<<ob.fill<<endl;
+		boxes.push_back(ob);
+	}
+	done:
+	std::cerr<<"GlExtra_OctreeCubes::postLoad: loaded "<<boxes.size()<<" boxes."<<std::endl;
+}
+
+void GlExtra_OctreeCubes::render(){
+	FOREACH(const OctreeBox& ob, boxes){
+		if(ob.fill<fillRangeDraw[0] || ob.fill>fillRangeDraw[1]) continue;
+		if(ob.level<levelRangeDraw[0] || ob.level>levelRangeDraw[1]) continue;
+		bool doFill=(ob.fill>=fillRangeFill[0] && ob.fill<=fillRangeFill[1] && (ob.fill!=0 || !noFillZero));
+		// -2: empty
+		// -1: recursion limit, empty
+		// 0: subdivided
+		// 1: recursion limit, full
+		// 2: full
+		Vector3r color=(ob.fill==-2?Vector3r(1,0,0):(ob.fill==-1?Vector3r(1,1,0):(ob.fill==0?Vector3r(0,0,1):(ob.fill==1)?Vector3r(0,1,0):(ob.fill==2)?Vector3r(0,1,1):Vector3r(1,1,1))));
+		glColor3v(color);
+		glPushMatrix();
+			glTranslatev(ob.center);
+			glScalef(2*ob.extents[0],2*ob.extents[1],2*ob.extents[2]);
+		 	if (doFill) glutSolidCube(1);
+			else glutWireCube(1);
+		glPopMatrix();
+	}
+}
+
+#endif /* YADE_OPENGL */
=== added file 'pkg/dem/DomainLimiter.hpp'
--- pkg/dem/DomainLimiter.hpp	1970-01-01 00:00:00 +0000
+++ pkg/dem/DomainLimiter.hpp	2013-10-15 05:01:48 +0000
@@ -0,0 +1,98 @@
+
+#include<yade/pkg/common/PeriodicEngines.hpp>
+#include<yade/core/PartialEngine.hpp>
+#include<yade/pkg/common/Sphere.hpp>
+
+class DomainLimiter: public PeriodicEngine{
+	public:
+		virtual void action();
+	YADE_CLASS_BASE_DOC_ATTRS(DomainLimiter,PeriodicEngine,"Delete particles that are out of axis-aligned box given by *lo* and *hi*.",
+		((Vector3r,lo,Vector3r(0,0,0),,"Lower corner of the domain."))
+		((Vector3r,hi,Vector3r(0,0,0),,"Upper corner of the domain."))
+		((long,nDeleted,0,Attr::readonly,"Cummulative number of particles deleted."))
+		((Real,mDeleted,0,,"Mass of deleted particles."))
+		((Real,vDeleted,0,,"Volume of deleted particles."))
+		((int,mask,-1,,"If mask is defined, only particles with corresponding groupMask will be deleted."))
+	);
+};
+REGISTER_SERIALIZABLE(DomainLimiter);
+
+class LawTester: public PartialEngine{
+	Body::id_t id1,id2; // shorthands for local use
+	public:
+		void init();
+		virtual void action();
+		void postLoad(LawTester&);
+		void warnDeprec(const string& s1, const string& s2){ if(!warnedDeprecPtRot){ warnedDeprecPtRot=true; LOG_WARN("LawTester."<<s1<<" is deprecated, use LawTester."<<s2<<" instead.");} }
+		Vector3r get_ptOurs(){ warnDeprec("ptOurs","uTest.head()"); return uTest.head<3>(); } Vector3r get_ptGeom(){ warnDeprec("ptGeom","uGeom.head()"); return uGeom.head<3>(); }
+		Vector3r get_rotOurs(){ warnDeprec("rotOurs","uTest.tail()"); return uTest.tail<3>(); }  Vector3r get_rotGeom(){ warnDeprec("rotGeom","uGeom.tail()"); return uGeom.tail<3>(); }
+	DECLARE_LOGGER;
+	YADE_CLASS_BASE_DOC_ATTRS_DEPREC_INIT_CTOR_PY(LawTester,PartialEngine,"Prescribe and apply deformations of an interaction in terms of local normal and shear displacements and rotations (using either :yref:`disPpath<LawTester.disPath>` and :yref:`rotPath<LawTester.rotPath>` [or :yref:`path<LawTester.path>` in the future]). Supported :yref:`IGeom` types are :yref:`ScGeom`, :yref:`L3Geom` and :yref:`L6Geom`. \n\nSee :ysrc:`scripts/test/law-test.py` for an example.",
+		((vector<Vector3r>,disPath,,Attr::triggerPostLoad,"Loading path, where each Vector3 contains desired normal displacement and two components of the shear displacement (in local coordinate system, which is being tracked automatically. If shorter than :yref:`rotPath<LawTester.rotPath>`, the last value is repeated."))
+		((vector<Vector3r>,rotPath,,Attr::triggerPostLoad,"Rotational components of the loading path, where each item contains torsion and two bending rotations in local coordinates. If shorter than :yref:`path<LawTester.path>`, the last value is repeated."))
+		((vector<string>,hooks,,,"Python commands to be run when the corresponding point in path is reached, before doing other things in that particular step. See also :yref:`doneHook<LawTester.doneHook>`. "))
+		((Vector6r,uGeom,Vector6r::Zero(),,"Current generalized displacements (3 displacements, 3 rotations), as stored in the interation itself. They should corredpond to :yref:`uTest<LawTester.uTest>`, otherwise a bug is indicated."))
+		((Vector6r,uTest,Vector6r::Zero(),,"Current generalized displacements (3 displacements, 3 rotations), as they should be according to this :yref:`LawTester`. Should correspond to :yref:`uGeom<LawTester.uGeom>`."))
+		((Vector6r,uTestNext,Vector6r::Zero(),Attr::hidden,"The value of uTest in the next step; since uTest is computed before uGeom is updated (in the next time step), uTest and uGeom would be always shifted by one timestep."))
+		((bool,warnedDeprecPtRot,false,Attr::hidden,"Flag to say that the user was already warned about using deprecated ptOurg/ptGeom/rotOurs/rotGeom."))
+		((Vector3r,shearTot,Vector3r::Zero(),Attr::hidden,"Current displacement in global coordinates; only used internally with ScGeom."))
+		((bool,displIsRel,true,,"Whether displacement values in *disPath* are normalized by reference contact length (r1+r2 for 2 spheres)."))
+		//((Vector3i,forceControl,Vector3i::Zero(),,"Select which components of path (non-zero value) have force (stress) rather than displacement (strain) meaning."))
+		((vector<int>,pathSteps,((void)"(constant step)",vector<int>(1,1)),Attr::triggerPostLoad,"Step number for corresponding values in :yref:`path<LawTester.path>`; if shorter than path, distance between last 2 values is used for the rest."))
+		((vector<int>,_pathT,,(Attr::readonly|Attr::noSave),"Time value corresponding to points on path, computed from *pathSteps*. Length is always the same as path."))
+		((vector<Vector6r>,_path,,(Attr::readonly|Attr::noSave),"Generalized displacement path values, computed from *disPath* and *rotPath* by appending zero initial value, and possibly repeating the last value to make lengths of *disPath* and *rotPath* match."))
+		((shared_ptr<Interaction>,I,,(Attr::hidden),"Interaction object being tracked."))
+
+		// axX, axY, axZ could be replaced by references to rows in trsf; perhaps do that in the future (in such a case, trsf would not be noSave)
+		((Vector3r,axX,,Attr::hidden,"Local x-axis in global coordinates (normal of the contact) |yupdate|"))
+		((Vector3r,axY,,Attr::hidden,"Local y-axis in global coordinates; perpendicular to axX; initialized arbitrarily, but tracked to be consistent. |yupdate|"))
+		((Vector3r,axZ,,(Attr::hidden|Attr::noSave),"Local z-axis in global coordinates; computed from axX and axY. |yupdate|"))
+		((Matrix3r,trsf,,(Attr::noSave|Attr::readonly),"Transformation matrix for the local coordinate system. |yupdate|"))
+		((size_t,_interpPos,0,(Attr::readonly|Attr::hidden),"State cookie for the interpolation routine."))
+		((Vector6r,uuPrev,Vector6r::Zero(),(Attr::readonly),"Generalized displacement values reached in the previous step, for knowing which increment to apply in the current step."))
+		((int,step,1,,"Step number in which this engine is active; determines position in path, using pathSteps."))
+		((string,doneHook,,,"Python command (as string) to run when end of the path is achieved. If empty, the engine will be set :yref:`dead<Engine.dead>`."))
+		((Real,renderLength,0,,"Characteristic length for the purposes of rendering, set equal to the smaller radius."))
+		((Real,refLength,0,(Attr::readonly),"Reference contact length, for rendering only."))
+		((Vector3r,contPt,Vector3r::Zero(),Attr::hidden,"Contact point (for rendering only)"))
+		((Real,idWeight,1,,"Float, usually ∈〈0,1〉, determining on how are displacements distributed between particles (0 for id1, 1 for id2); intermediate values will apply respective part to each of them. This parameter is ignored with 6-DoFs :yref:`IGeom`."))
+		((Real,rotWeight,1,,"Float ∈〈0,1〉 determining whether shear displacement is applied as rotation or displacement on arc (0 is displacement-only, 1 is rotation-only). Not effective when mutual rotation is specified."))
+		// reset force components along individual axes, instead of blocking DOFs which have no specific direction (for the force control)
+		, /* deprec */ ((path,disPath,"LawTester.path will be used for generalized displacement (6-component) loading path in the future."))
+		, /* init */
+		, /* ctor */
+		, /* py */ .add_property("ptOurs",&LawTester::get_ptOurs,"first 3 components of uTest |ydeprecated|") .add_property("ptGeom",&LawTester::get_ptGeom,"first 3 components of uGeom |ydeprecated|") .add_property("rotOurs",&LawTester::get_rotOurs,"last 3 components of uTest |ydeprecated|") .add_property("rotGeom",&LawTester::get_rotGeom,"last 3 components of uGeom |ydeprecated|")
+	);
+};
+REGISTER_SERIALIZABLE(LawTester);
+
+#ifdef YADE_OPENGL
+#include<yade/pkg/common/OpenGLRenderer.hpp>
+
+class GlExtra_LawTester: public GlExtraDrawer{
+	public:
+	DECLARE_LOGGER;
+	virtual void render();
+	YADE_CLASS_BASE_DOC_ATTRS(GlExtra_LawTester,GlExtraDrawer,"Find an instance of :yref:`LawTester` and show visually its data.",
+		((shared_ptr<LawTester>,tester,,,"Associated :yref:`LawTester` object."))
+	);
+};
+REGISTER_SERIALIZABLE(GlExtra_LawTester);
+
+class GlExtra_OctreeCubes: public GlExtraDrawer{
+	public:
+	struct OctreeBox{ Vector3r center, extents; int fill; int level; };
+	std::vector<OctreeBox> boxes;
+	void postLoad(GlExtra_OctreeCubes&);
+	virtual void render();
+	YADE_CLASS_BASE_DOC_ATTRS(GlExtra_OctreeCubes,GlExtraDrawer,"Render boxed read from file",
+		((string,boxesFile,,Attr::triggerPostLoad,"File to read boxes from; ascii files with ``x0 y0 z0 x1 y1 z1 c`` records, where ``c`` is an integer specifying fill (0 for wire, 1 for filled)."))
+		((Vector2i,fillRangeFill,Vector2i(2,2),,"Range of fill indices that will be filled."))
+		((Vector2i,fillRangeDraw,Vector2i(-2,2),,"Range of fill indices that will be rendered."))
+		((Vector2i,levelRangeDraw,Vector2i(-2,2),,"Range of levels that will be rendered."))
+		((bool,noFillZero,true,,"Do not fill 0-fill boxed (those that are further subdivided)"))
+	);
+};
+REGISTER_SERIALIZABLE(GlExtra_OctreeCubes);
+#endif
+