I'm working on building Cppcheck on AIX with the xlC
compiler (see previous question). Checker classes all derive from a Check
class, whose constructor registers each object in a global list:
check.h
class Check {
public:
Check() {
instances().push_back(this);
instances().sort();
}
static std::list<Check *> &instances();
virtual std::string name() const = 0;
private:
bool operator<(const Check *other) const {
return (name() < other->name());
}
};
checkbufferoverrun.h
class CheckBufferOverrun: public Check {
public:
// ...
std::string name() const {
return "Bounds checking";
}
};
The problem I appear to be having is with the instances().sort()
call. sort()
will call Check::operator<()
which calls Check::name()
on each pointer in the static instances()
list, but the Check
instance that was just added to the list has not yet had its constructor fully run (because it's still inside Check::Check()
). Therefore, it should be undefined behaviour to call ->name()
on such a pointer before the CheckBufferOverrun
constructor has completed.
Is this really undefined behaviour, or am I missing a subtlety here?
Note that I don't think the call to sort()
is strictly required, but the effect is that Cppcheck runs all its checkers in a deterministic order. This only affects the output in the order in which errors are detected, which causes causes some test cases to fail because they're expecting the output in a particular order.
Update: The question as above still (mostly) stands. However, I think the real reason why the call to sort()
in the constructor wasn't causing problems (ie. crashing by calling a pure virtual function) is that the Check::operator<(const Check *)
is never actually called by sort()
! Rather, sort()
appears to compare the pointers instead. This happens in both g++
and xlC
, indicating a problem with the Cppcheck code itself.
Yes, it's undefined. The standard specifically says so in 10.4/6