Help interpreting load time error

I have started to customize the robot template for our team. I have sub-classed the SimpleRobot class to add some of our custom functions. Everything compiled and the code was downloaded into the cRIO fine. But when I rebooted the robot, at the end of the boot, it gave me the following output and I have no idea why TrcRobot is undefined. TrcRobot is my subclass of SimpleRobot. Simplified code construct is shown below. I am expecting to see compiler errors if I made mistakes. But the compiler didn’t complain and instead I got load time error of undefined symbol. Can somebody tell me what it means? Thanks.


FRC_NetworkCommunication was compiled from SVN revision 2064
FPGA Hardware GUID: 0xAD9A5591CC64E4DF756D77D1B57A549E
FPGA Software GUID: 0xAD9A5591CC64E4DF756D77D1B57A549E
FPGA Hardware Version: 2010
FPGA Software Version: 2010
FPGA Hardware Revision: 1.3.0
FPGA Software Revision: 1.3.0
* Loading StartupDlls: FRC_UserProgram
Warning: module 0x134c5d0 (FRC_UserProgram.out) holds reference to undefined symbol _ZTV8TrcRobot.
Warning: module 0x134c5d0 (FRC_UserProgram.out) holds reference to undefined symbol _ZTI8TrcRobot.
(unloading partially loaded module FRC_UserProgram.out)
...FRC_UserProgram failed to load.


class TrcRobot: public SimpleRobot
{
public:
    virtual void SomeCustomizedMethods();
    void Autonomous()
    {
        //Autonomous main loop.
        SomeCustomizedMethods();
    }
    void OperatorControl()
    {
        //Operator control main loop.
        SomeCustomizedMethods();
    }
};

class MyRobot: public TrcRobot
{
    MyRobot()
    {
        //Instantiate other objects.
    }
    ~MyRobot()
    {
        //Destroy objects created in constructor.
    }
    void SomeCustomizedMethods()
    {
        //Autonomous or Operator Control tasks.
    }
}
START_ROBOT_CLASS(MyRobot);

I have a few things:

  • This is only slightly related to your problem, but if you’re going to implement an Autonomous() or OperatorControl() in MyRobot, you’ll need to make these functions virtual in TrcRobot. The deconstructor for MyRobot (MyRobot::~MyRobot) also won’t be called unless you make a virtual deconstructor in TrcRobot. Make sure to check inheritence for stuff like this.

  • An “undefined symbol” basically means that the program can’t find a function or piece of code where it thinks there should be one. This could be caused by different things:
    [List]

  • A header file contains the definition of a function, and the source files either don’t have the function at all or the function has a different definition in the source files. (This is usually caught during compile-time though).

  • Your versions are off. The code’s looking for something in the WPI library where it shouldn’t be. Make sure both your robot and your WindRiver workbench have the newest updates.

[/LIST]

Taking a second look at it, though, I thought of this:

Since the undefined symbol is “_ZTV8TrcRobot.”, I’d guess this is caused by neither of the above possibilities, but something involving inheritance. Put constructors in both TrcRobot and MyRobot (you only have one in MyRobot right now). The START_ROBOT_CLASS macro instantiates the class you give it, which will call not just call the base constructor SimpleRobot::SimpleRobot() but also TrcRobot::TrcRobot() and MyRobot::MyRobot(). I’m guessing that after calling the SimpleRobot constructor it looks for the TrcRobot constructor and freaks out when there isn’t one.

Let me know if this fixes the problem.

Yes, I have constructors on both MyRobot and TrcRobot (sorry that I simplified too much and did not show them in my code example). I suspect about inheritance too. So if the undefined symbol means it could not find the constructor of TrcRobot, I am puzzled because it’s there. I am going to cut everything non-essential out to isolate the problem tonight and see what it is really complaining.

Cool. If I can help any more, feel free to post here or PM me.

The reason the compiler (it’s actually the linker that would complain) is not giving you an error is that VxWorks links at load time instead of requiring a stub library for external linkage. Any symbols that are not defined when building are simply assumed to exist in the environment that you program will load in. If that does not hold true, you get errors from the loader.

I see. That makes sense now. So according to the linker error, am I missing the constructor of TrcRobot then?

No… it’s not the constructor. It’s the vtable and the typeinfo.

If you run the tool C:\WindRiver\gnu\3.4.4-vxworks-6.3\x86-win32\bin\c++filtppc.exe and pass in the mangled name, it will tell you what it is in plain text.

Yes, you are right, it is the typeinfo and vtable. But what does it mean if these symbols are missing? TrcRobot is the base class of MyRobot. Why would the vtable of the TrcRobot be missing?

You have not included the actual code, so it’s a little hard to say, but I would speculate that you don’t have a virtual destructor defined for TrcRobot. Since the RobotBase and SimpleRobot destructors are virtual and TrcRobot inherits from them and is then inherited from by MyRobot, you must have a virtual destructor at all levels of the hierarchy. Leaving it out of one class in the middle won’t work. The default destructor is not virtual.

-Joe

Yes, I found it out yesterday. I thought I had an empty destructor defined but I missed that one. Thanks for the help.