Skip to content

Commit 01d4c46

Browse files
quaglacopybara-github
authored andcommitted
Store the mjsCompiler -> appended mjSpec map when appending an mjSpec.
Previously, we stored the source `mjSpec` during a copy as a hack for having access to the compiler options, but this is not robust since we cannot guarantee that 1) the source `mjSpec` is not destroyed before we need to look up the compiler options nor 2) that the `mjSpec` was appended without a copy. While 2) could be solved by simply handling an additional case in `mjCModel::FindSpec`, using a map also solves 1) and it is easier to understand. PiperOrigin-RevId: 741504264 Change-Id: Iab1bfd9e61299a94fa8caf3a244c067b09d54384
1 parent 6e19035 commit 01d4c46

File tree

4 files changed

+69
-25
lines changed

4 files changed

+69
-25
lines changed

src/user/user_model.cc

+14-15
Original file line numberDiff line numberDiff line change
@@ -212,9 +212,6 @@ mjCModel::mjCModel() {
212212
// create mjCBase lists from children lists
213213
CreateObjectLists();
214214

215-
// the source spec is the model itself, overwritten in the copy constructor
216-
source_spec_ = &spec;
217-
218215
// set the signature
219216
spec.element->signature = 0;
220217
}
@@ -223,7 +220,6 @@ mjCModel::mjCModel() {
223220

224221
mjCModel::mjCModel(const mjCModel& other) {
225222
CreateObjectLists();
226-
source_spec_ = (mjSpec*)&other.spec;
227223
*this = other;
228224
}
229225

@@ -239,6 +235,7 @@ mjCModel& mjCModel::operator=(const mjCModel& other) {
239235
// copy attached specs first so that we can resolve references to them
240236
for (const auto* s : other.specs_) {
241237
specs_.push_back(mj_copySpec(s));
238+
compiler2spec_[&s->compiler] = specs_.back();
242239
}
243240

244241
// the world copy constructor takes care of copying the tree
@@ -1165,9 +1162,13 @@ mjCPlugin* mjCModel::AddPlugin() {
11651162

11661163

11671164
// append spec to spec
1168-
void mjCModel::AppendSpec(mjSpec* spec) {
1165+
void mjCModel::AppendSpec(mjSpec* spec, const mjsCompiler* compiler_) {
11691166
// TODO: check if the spec is already in the list
11701167
specs_.push_back(spec);
1168+
1169+
if (compiler_) {
1170+
compiler2spec_[compiler_] = spec;
1171+
}
11711172
}
11721173

11731174

@@ -1461,10 +1462,15 @@ mjSpec* mjCModel::FindSpec(std::string name) const {
14611462

14621463

14631464
// find spec by mjsCompiler pointer
1464-
mjSpec* mjCModel::FindSpec(const mjsCompiler* compiler_) const {
1465-
if (&GetSourceSpec()->compiler == compiler_) {
1466-
return (mjSpec*)&spec;
1465+
mjSpec* mjCModel::FindSpec(const mjsCompiler* compiler_) {
1466+
if (compiler_ == &spec.compiler) {
1467+
return &spec;
1468+
}
1469+
1470+
if (compiler2spec_.find(compiler_) != compiler2spec_.end()) {
1471+
return compiler2spec_[compiler_];
14671472
}
1473+
14681474
for (auto s : specs_) {
14691475
mjSpec* source = static_cast<mjCModel*>(s->element)->FindSpec(compiler_);
14701476
if (source) {
@@ -1476,13 +1482,6 @@ mjSpec* mjCModel::FindSpec(const mjsCompiler* compiler_) const {
14761482

14771483

14781484

1479-
// get the spec from which this model was created
1480-
mjSpec* mjCModel::GetSourceSpec() const {
1481-
return source_spec_;
1482-
}
1483-
1484-
1485-
14861485
//------------------------------- COMPILER PHASES --------------------------------------------------
14871486

14881487
// make lists of objects in tree: bodies, geoms, joints, sites, cameras, lights

src/user/user_model.h

+5-8
Original file line numberDiff line numberDiff line change
@@ -215,7 +215,9 @@ class mjCModel : public mjCModel_, private mjSpec {
215215
mjCTuple* AddTuple();
216216
mjCKey* AddKey();
217217
mjCPlugin* AddPlugin();
218-
void AppendSpec(mjSpec* spec);
218+
219+
// append spec to this model, optionally map compiler options to the appended spec
220+
void AppendSpec(mjSpec* spec, const mjsCompiler* compiler = nullptr);
219221

220222
// delete elements marked as discard=true
221223
template <class T> void Delete(std::vector<T*>& elements,
@@ -248,7 +250,7 @@ class mjCModel : public mjCModel_, private mjSpec {
248250
mjCBase* FindObject(mjtObj type, std::string name) const; // find object given type and name
249251
mjCBase* FindTree(mjCBody* body, mjtObj type, std::string name); // find tree object given name
250252
mjSpec* FindSpec(std::string name) const; // find spec given name
251-
mjSpec* FindSpec(const mjsCompiler* compiler_) const; // find spec given mjsCompiler
253+
mjSpec* FindSpec(const mjsCompiler* compiler_); // find spec given mjsCompiler
252254
void ActivatePlugin(const mjpPlugin* plugin, int slot); // activate plugin
253255

254256
// accessors
@@ -316,9 +318,6 @@ class mjCModel : public mjCModel_, private mjSpec {
316318
// map from default class name to default class pointer
317319
std::unordered_map<std::string, mjCDef*> def_map;
318320

319-
// get the spec from which this model was created
320-
mjSpec* GetSourceSpec() const;
321-
322321
// set deepcopy flag
323322
void SetDeepCopy(bool deepcopy) { deepcopy_ = deepcopy; }
324323

@@ -332,9 +331,6 @@ class mjCModel : public mjCModel_, private mjSpec {
332331
// settings for each defaults class
333332
std::vector<mjCDef*> defaults_;
334333

335-
// spec from which this model was created in copy constructor
336-
mjSpec* source_spec_;
337-
338334
// list of active plugins
339335
std::vector<std::pair<const mjpPlugin*, int>> active_plugins_;
340336

@@ -453,5 +449,6 @@ class mjCModel : public mjCModel_, private mjSpec {
453449
bool deepcopy_; // copy objects when attaching
454450
bool attached_ = false; // true if model is attached to a parent model
455451
int uid_count_ = 0; // unique id count for all objects
452+
std::unordered_map<const mjsCompiler*, mjSpec*> compiler2spec_; // map from compiler to spec
456453
};
457454
#endif // MUJOCO_SRC_USER_USER_MODEL_H_

src/user/user_objects.cc

+2-2
Original file line numberDiff line numberDiff line change
@@ -904,7 +904,7 @@ mjCBody& mjCBody::operator+=(const mjCBody& other) {
904904
mjCBody& mjCBody::operator+=(const mjCFrame& other) {
905905
// append a copy of the attached spec
906906
if (other.model != model && !model->FindSpec(mjs_getString(other.model->spec.modelname))) {
907-
model->AppendSpec(mj_copySpec(&other.model->spec));
907+
model->AppendSpec(mj_copySpec(&other.model->spec), &other.model->spec.compiler);
908908
}
909909

910910
// create a copy of the subtree that contains the frame
@@ -2038,7 +2038,7 @@ mjCFrame& mjCFrame::operator=(const mjCFrame& other) {
20382038
mjCFrame& mjCFrame::operator+=(const mjCBody& other) {
20392039
// append a copy of the attached spec
20402040
if (other.model != model && !model->FindSpec(mjs_getString(other.model->spec.modelname))) {
2041-
model->AppendSpec(mj_copySpec(&other.model->spec));
2041+
model->AppendSpec(mj_copySpec(&other.model->spec), &other.model->spec.compiler);
20422042
}
20432043

20442044
// apply namespace and store keyframes in the source model

test/user/user_api_test.cc

+48
Original file line numberDiff line numberDiff line change
@@ -2514,6 +2514,54 @@ TEST_F(MujocoTest, DifferentUnitsAllowed) {
25142514
mj_deleteModel(copied_model);
25152515
}
25162516

2517+
TEST_F(MujocoTest, DifferentOptionsInAttachedFrame) {
2518+
static constexpr char xml_parent[] = R"(
2519+
<mujoco>
2520+
<worldbody/>
2521+
</mujoco>
2522+
)";
2523+
2524+
static constexpr char xml_child[] = R"(
2525+
<mujoco>
2526+
<compiler eulerseq="zyx"/>
2527+
<worldbody>
2528+
<frame name="child" >
2529+
<site euler="0 90 180"/>
2530+
</frame>
2531+
</worldbody>
2532+
</mujoco>
2533+
)";
2534+
2535+
// load specs and compile child
2536+
mjSpec* parent = mj_parseXMLString(xml_parent, 0, nullptr, 0);
2537+
EXPECT_THAT(parent, NotNull());
2538+
mjSpec* child = mj_parseXMLString(xml_child, 0, nullptr, 0);
2539+
EXPECT_THAT(child, NotNull());
2540+
mjModel* m_child = mj_compile(child, 0);
2541+
EXPECT_THAT(m_child, NotNull());
2542+
2543+
// attach child frame to parent worldbody
2544+
mjsBody* world = mjs_findBody(parent, "world");
2545+
EXPECT_THAT(world, NotNull());
2546+
mjsFrame* child_frame = mjs_findFrame(child, "child");
2547+
EXPECT_THAT(child_frame, NotNull());
2548+
mjsFrame* attached_frame = mjs_attachFrame(world, child_frame, "child-", "");
2549+
EXPECT_THAT(attached_frame, NotNull());
2550+
2551+
// wrap the child frame in the parent frame and compile
2552+
mjModel* m_attached = mj_compile(parent, 0);
2553+
EXPECT_THAT(m_attached, NotNull());
2554+
EXPECT_NEAR(m_attached->site_quat[0], m_child->site_quat[0], 1e-6);
2555+
EXPECT_NEAR(m_attached->site_quat[1], m_child->site_quat[1], 1e-6);
2556+
EXPECT_NEAR(m_attached->site_quat[2], m_child->site_quat[2], 1e-6);
2557+
EXPECT_NEAR(m_attached->site_quat[3], m_child->site_quat[3], 1e-6);
2558+
2559+
mj_deleteSpec(parent);
2560+
mj_deleteSpec(child);
2561+
mj_deleteModel(m_child);
2562+
mj_deleteModel(m_attached);
2563+
}
2564+
25172565
TEST_F(MujocoTest, CopyAttachedSpec) {
25182566
static constexpr char xml_parent[] = R"(
25192567
<mujoco>

0 commit comments

Comments
 (0)