Sunday, February 24, 2013

The Apprentice NAOmancer (NAO tutorial part 3)

NAO Linux C++ development cheat sheet / tutorial, Part 3

In Part 2, we made our own simple application that runs on your computer, connects to NAO and displays every second on the filtered torso angle with respect to the world.

In this part, we transform this app into a module that can run on the robot and record torso angles in a file.

You can browse and clone the repository containing the source corresponding to this project on Bitbucket. The tag corresponding to this part of the tutorial is v0.1.

Creating and opening the NAOqi module project


Let's make a fresh project.

cd ~/nao/workspace
qibuild create inmon # [in]ertial sensor [mon]itor. catchy huh? 
cd inmon

We won't really use the default source files, so let's delete all source files from the project:

rm *.cpp

And we directly create the files we need and edit CMakeLists.txt to reflect that, because Qt Creator likes it that way (maybe there is an easier way? I usually don't use Qt Creator but Emacs, so I don't know if there is no easier way...).

touch inmon.{h,c}pp naoqi_module_loader.cpp inmon_client.cpp

Dimitri tells me that modern versions of Qt Creator will just work fine and update the project when CMakeLists.txt is edited and saved, so you might not need to fight with your IDE by reloading it each time you add or move a file. So, don't do like me, install the latest Qt Creator, not your distro's default one.

CMakeLists.txt

cmake_minimum_required(VERSION 2.8)
project(inmon)
find_package(qibuild)

# The naoqi module
qi_create_lib(inmon SHARED inmon.cpp naoqi_module_loader.cpp SUBFOLDER naoqi)
qi_use_lib(inmon ALCOMMON BOOST)
qi_stage_lib(inmon)

# Create an executable that can call the inmon module (locally or remotely)
qi_create_bin(inmon_client inmon_client.cpp)
qi_use_lib(ALCOMMON)

Configure the project, as this creates the build directory that Qt Creator wants (even if we will only use it as an editor, it wants a build directory):

qibuild configure

There is a command to create Qt Creator project files that will work with qiBuild:

qibuild open

with that command, you are now in QT Creator, and it's asking you about the build directory. Use ~/nao/workspace/inmon/build-atom as build directory, and click 'run cmake', and 'finish'. Your project is now open in Qt Creator, and the source files should be available for editing.

We already know how to get the filtered inertial sensor values, so let's focus on all the code that is around that. In a first iteration of this module, we will examine:

  • the actual class (IntertialMonitor) that represents our module,
  • the code for loading and unloading that module into NAOqi,
  • making some of the member functions of our IntertialMonitor class available through NAOqi so that they can be called by an external program or another module.
  • making the module record the inertial sensor values to a file on NAO
  • making a client program that lets us call the exported functions of our module (start, stop, configure) from a remote computer or from NAO directly


Programming a local (on-robot) NAOqi module


Module loading code in naoqi_module_loader.cpp


A C++ NAOqi module is a library with some specific functions available. So, it's C++ code that has been compiled as a library (using qi_create_lib in the CMakeLists.txt file). Two special functions have to be there to let NAOqi load and unload the module:

- _createModule that sets the broker and creates an instance (an object)
- _closeModule that does what it says.

I've separated those two functions in a file called naoqi_module_loader.cpp. They could also be in inmon.cpp along with the module class, but I find it clearer if they're not in the same file.

In practice, this code will rarely vary, you're going to copy-paste naoqi_module_loader.cpp from one local module project to the other. However, I've simplified it a little, removing code that lets you make the module a remote module (that can be loaded and used on a separate computer with its own local broker connected to NAO's broker over the network). For a full example of a module loader that can be used for both local and remote modules, see Aldebaran's loader there.

The NAOqi module class header inmon.hpp


The header file inmon.hpp contains the declaration of the InternalMonitor class. It's a very useful to first look at that file to have an overview of the class without implementation details. Besides the include statements and other trivial things, the header file contains the following:


  1. class InertialMonitor : public AL::ALModule {  
  2. public:  
  3.   InertialMonitor(boost::shared_ptr<AL::ALBroker> pBroker, const std::string& pName);  
  4.   ~InertialMonitor();  
  5.   void init();  
  6.   void exit();  
  7.   void startMonitor(int milli);  
  8.   void startMonitorAndRecord(int milli, const std::string file);  
  9.   void stopMonitor();  
  10. private:  
  11.   struct Impl;  
  12.   boost::shared_ptr<Impl> impl;  
  13. };  


The first four public members are not so interesting, they are similar for all NAOqi module classes. When NAOqi  instantiates the class, the constructor (InertialMonitor(shared_ptr, string)) is called, followed by the init() method. ~InertialMonitor is the object destructor. Unloading the module calls exit() prior to destruction. Note that those four methods should have been qualified by the keyword virtual if we had planned to make derived classes from InertialMonitor. You can read more about virtual methods in C++ on the very useful C++ FAQ website.

startMonitor(int), startMonitorAndRecord(int, string) and stopMonitor() are functions that we want to make available externally and to other modules.

- startMonitor(int) and startMonitorAndRecord(int, string) start or restart the inertial sensor monitor with the given sampling period, recording the data locally in the given plain text output file if provided (the data are appended to the file if it already exists).
- stopMonitor() stops the monitor and closes the recording file if any.

The private section of the class might seem a little obscure:


  1. private:  
  2.   struct Impl;  
  3.   boost::shared_ptr<Impl> impl; 


Normally, the private members of a class are all data attributes and member functions that are here for the convenience of the implementer, and not to be accessed by anyone but the class itself. So, we might expect to see something revealing the internals of the module, for instance like that:


  1. public:  
  2.   void operator()(); // thread method. must be public to be callable, I think.  
  3. private:  
  4.   boost::shared_ptr<AL::ALMemoryProxy> memoryProxy;  
  5.   boost::posix_time::time_duration samplingPeriod;   
  6.   boost::thread *t;  // thread of the sampling code  
  7.   bool stopThread;   // flag to stop the thread, also condVar's timed_wait predicate against spurious unblocks  
  8.   boost::mutex stopThreadLock; // lock  
  9.   boost::condition_variable condVar; // mechanism to control the thread's execution  
  10.   // … probably more stuff  


But instead, all we see is an opaque struct of type Impl, and a pointer impl to one instance of Impl. This is typical of the PImpl (means "[P]rivate [Impl]ementation") idiom, which is very useful in C++ to more clearly separate the public part of a class from its internal implementation. I advise to follow that idiom whenever possible. Here, all the threading mess still exists, but only in the cpp file.

The most obvious advantage of PImpl is that the header files are completely insensitive to implementation changes. So, for instance, if I decide to change from a wait mechanism that is internal to the class to some sort of event-driven (external timer maybe?) mechanism, well all these private attributes are certainly going to change, but if they are hidden behind a struct that is only defined in the cpp file, the header will remain exactly the same.

The NAOqi module class implementation inmon.cpp


Now that we have seen the value of the opaque pointer to the implementation, we need to actually define this class and that Impl structure. inmon.cpp contains the whole implementation. This section discusses some parts of it.

The functions of the outer InertialMonitor class (ctor, dtor, init, exit, startMonitor, startMonitorAndRecord, stopMonitor) should make use of the members of the inner PImpl structure (InertialMonitor::Impl).

Overview

(the snippets below in the Overview section are pseudo-C++ code, simplified for readabililty.)
I have chosen to write the Impl structure with the following members:


  1. struct InertialMonitor::Impl {  
  2.   boost::shared_ptr<AL::ALMemoryProxy> memoryProxy;  
  3.   bool stopThread; // flag to stop the thread, and predicate against spurious unblocks  
  4.   std::ofstream outputFile;  
  5.   boost::posix_time::time_duration samplingPeriod;  
  6.   
  7.   Impl(InertialMonitor &mod);  
  8.   ~Impl();  
  9.   void setupOutputFile(const std::string filename);  
  10.   void closeOutputFile();  
  11.   void operator()();  
  12. }


The member function operator() makes Impl instances callable, and it's the function that should contains the sampling loop; something like the following pseudocode:


  1. void InertialMonitor::Impl::operator()() {  
  2.   while (!stopThread) {  
  3.     outputFile << timestamp << ", " << x << ", " << y << "\n";  
  4.     wait(timeout);  
  5.   }  
  6. }  


The functions setupOutputFile() and closeOutputFile() open the InternalMonitor record file and close it. They have to ensure that at all times, at most one file is open.

Our module has to be able to do two things simultaneously: respond to requests from clients and other modules, and monitor the inertial sensors at a certain frequency, maybe recording the observations in some file. The easiest way to do that is to have (when required) a thread T2 that does the sampling and recording, and another thread T1 that processes the remote calls and controls T2 accordingly.

Let's see a use case:

On the robot, NAOqi has loaded the InertialMonitor module, so it has created an instance of it that we will call m, and it sits idle, waiting for a RPC.

On a remote computer, I call p.startMonitor(1000) (I call the member function startMonitor on a proxy p of the InertialMonitor module), indicating that I want the InertialMonitor module to sample the inertial sensor every second, without recording the output. The computer's instance of naoqi (the "remote broker") communicates with naoqi on the robot (local broker) to make that call.

On NAO, the module's main thread of execution (T1), which was waiting to be activated by such a request, starts running through m.startMonitor, the method of the local InertialMonitor instance m that corresponds to the method called on the remote proxy module. In short, T1 runs through startMonitor on the robot.

At some point the code of startMonitor, T1 spawns a thread T2 that runs through an infinite loop. Meanwhile, T1 finishes running through startMonitor and "goes back to wait" for another request.

T2 continues its loop: every second, it reads the inertial monitor sensor values from the local ALMemory instance, and when done, goes back to wait for the next second, or to be interrupted by a signal from T1. It would also write the values to the file, had it been requested (using startMonitorAndRecord).

On the remote computer, I call p.startMonitorAndRecord(100, "/home/nao/test"). I want a 100 ms sampling, recorded in the file test in NAO's home folder (on the robot). The remote broker sends that call to the local broker.

The local naoqi runs T1 through m.startMonitorAndRecord, with the parameters I gave. Note that at that time, T2 should still exist from the previous call, and it is either running (maybe busy reading sensor values), or waiting for the next sampling time. But we now want to have this monitoring thread run 10 times faster and record the readings in a file. So we could go two ways about it: communicate these new parameters to T2 and make it adapt its behavior to match that, or terminate T2 and make a new thread with the new parameters.

In programming, simplicity is preferable when performance is not critical. In the present case, that means that just stopping the execution of the sensor capture / recording loop and restarting it with the new parameters is better because it's simpler.

So, in the execution of m.startMonitorAndRecord by T1, we have to first shutdown T2, then spawn it again with the new parameters. Of course, we can only shut down T2 when it is not in the middle of something critical, like writing to the record file, or reading the sensor values. In fact, it is only safe to stop T2 in the beginning of the loop. However, in our example here, T2 could be waiting up to 1 second to reach the beginning of the loop again and notice "the request to terminate" of T1. Meanwhile, T1 would be waiting for T2, and I, the user on the remote computer, would be wondering why my command to increase the sampling frequency and record it on file is still not executed. That means that T1 should be able to notify a "sleeping" T2 that he has to wake up now, because it's "time to die". In consequence, the "sleep" of T2 would better be the last thing it does before testing if it's been awaken by a request to terminate.

Back to our scenario, T2 is awaken from its wait by T1's signal, and detects that request to finish, so it exits its 1-second sampling loop and finishes. T1, which was waiting for this end of T2, continues by spawning another, new thread T2 that runs through the same infinite loop, but now configured for a 100ms period, and to record to the given file. T1 finishes running through startMonitorAndRecord and "goes back to wait" for another request.

When the request to end T2 comes, this time, we have to also ensure that the recording file is properly closed.

Threading


From that overview, let's summarise the main points of the concurrent execution of T1 and T2, and see how it's done in the code:

The main execution thread (T1) is assumed to always exist, but its state is managed by the local instance of NAOqi. In other words, we ono need to know that when the InertialMonitor module is loaded by NAOqi, 2 function are called: the constructor, and init(). Then, the module "waits" for one of its public API methods to be called through NAOqi. When such a method is called, T1 runs through it and can spawn new threads that will run independently if necessary. But T1 itself should finish running the method and "go back to wait" to another NAOqi call.

When launching a new sampling thread, T1 has to first ensure that any bother sampling thread is done. If there is a T2 running, that means:


  • send a signal to T2 to terminate
  • wait that T2 is stopped
  • close the output file is any (using Impl::closeOutputFile)


Once this is done, T1 has to:


  • open the new output file in append mode (we don't want to overwrite old records)
  • configure the sampling period (Impl:: samplingPeriod)
  • start the new thread


On its side, T2 has the following responsibilities:


  • sample the inertial sensor values at the frequency required, without drift.
  • when notified of a request to self-terminate, 
    • if waiting: stop waiting for the next sampling period, and go right away to terminate.
    • if doing something else: wrap it up cleanly, and go to terminate.


In the code, that is implemented in the following way:


  1. void InertialMonitor::Impl::operator()()  {  
  2.   boost::unique_lock<boost::mutex> fileLock(outputFileLock); // starts locked  
  3.   bool stopThreadCopy;  
  4.   boost::posix_time::ptime time_t_epoch(boost::gregorian::date(1970,1,1));  
  5.   float x, y;  
  6.   stopThreadLock.lock();  
  7.   stopThreadCopy = stopThread;  
  8.   stopThreadLock.unlock();  
  9.   float *intertialSensorX =  
  10.       static_cast<float*>(memoryProxy->getDataPtr(intertialSensorXKey));  
  11.   float *intertialSensorY =  
  12.       static_cast<float*>(memoryProxy->getDataPtr(intertialSensorYKey));  
  13.   boost::system_time tickTime = boost::get_system_time();  
  14.   while (!stopThreadCopy) {  
  15.     // get values  
  16.     x = *intertialSensorX;  
  17.     y = *intertialSensorY;  
  18.     // note the time (in nanoseconds since Jan 1 1970)  
  19.     boost::posix_time::time_duration duration = boost::get_system_time() - time_t_epoch;  
  20.     long long timestampNanos = duration.total_nanoseconds();  
  21.     // record them  
  22.     if (outputFile_p) {  
  23.       try {  
  24.         outputFile << timestampNanos << ", " << x << ", " << y << "\n"// endl flushes  
  25.       } catch (std::exception& e) {  
  26.         qiLogError("InertialMonitor") << "Failed to write \"" <<  timestampNanos << ", "  
  27.                                       << x << ", " << y << "\" to the output file: "  
  28.                                       << e.what() << std::endl;  
  29.       }  
  30.     }  
  31.     // calculate next tick time  
  32.     tickTime += samplingPeriod;  
  33.     // wait for timeout, unlock fileLock while waiting, and control for spurious wakes  
  34.     // by checking stopThread.  
  35.     condVar.timed_wait(fileLock, tickTime, boost::lambda::var(stopThread));  
  36.     stopThreadLock.lock();  
  37.     stopThreadCopy = stopThread;  
  38.     stopThreadLock.unlock();  
  39.   }  
  40. }  


Note the lock/read/unlock and lock/write/unlock operations in the above code for all variables that could potentially be accessed by T1 concurrently. Let's take for instance the 3 instructions stopThreadLock.lock(); stopThreadCopy = stopThread; stopThreadLock.unlock(); This means that before reading the value of the variable stopThread, T2 has to acquire the lock named stopThreadLock (a member of Impl of type boost::mutex). Once the value is read, T2 releases the lock on stopThread. stopThread is the variable used to signal to T2 that it has to stop, so T1 might write in this variable anytime. So, in all places where this stopThread variable is manipulated, it is surrounded by the lock/unlock mechanism. T2 will only be able to acquire this lock if another thread (here, T1) does not already hold it. For instance, in InertialMonitor::stopMonitor():


  1. void InertialMonitor::stopMonitor() {  
  2.   try {  
  3.     impl->closeOutputFile();  
  4.     impl->stopThreadLock.lock();  
  5.     impl->stopThread = true;  
  6.     impl->stopThreadLock.unlock();  
  7.     // wake up, it time ...  
  8.     impl->condVar.notify_one();  
  9.     // ... to die  
  10.     if (impl->t) {  
  11.       impl->t->join();  
  12.     }  
  13.   } catch (std::exception& e) {  
  14.     qiLogError("InertialMonitor") << "Failed to stop intertial monitor: "  
  15.                         << e.what() << std::endl;  
  16.     exit();  
  17.   }  
  18. }  


This locking mechanism is a primitive way to do concurrent programming, but it's sufficient for our purpose.

In the above code for stopMonitor, we also see how notification to T2 to stop its activity is implemented:


  • impl->stopThread is set to true
  • T2 is awaken from its sleep through a signal (impl->condVar.notify_one())
  • T1 waits for T2 to finish and die (impl->t->join())


File handling


We need to record all samples, with for each sample: the time at which it was taken, the value of the inertial sensor for the X axis, and the value for Y. But that's not enough to know that: a decision has to be made regarding the format of the record files.


  • very compressed binary format vs text files? 
  • format for dates?
  • format for sensor values?


It's better to have human-readable text, but it's also necessary not to loose precision in the output, and to ensure that the logs are also easily machine-readable. The sensor values come as floats, we can record them as they would be printed, as there is noise in the sensor readings and trailing digits are just garbage.

The format for dates is the number of nanoseconds since January 1st, 1970 (Unix "time 0") on NAO's clock. I choose an absolute time because it's generally more informative at the cost of a little more space and clutter, and because the file may be appended during a subsequent call to a recording function or even a subsequent run of the module. A relative clock may be reset by that, and the records would be inconsistent, with a rollback to time 0 in the middle of the file.

Finally, what should happen when the same file path is given during several subsequent calls of the startMonitorAndRecord function? We have the choice between failing with an error message, overwriting the previous file, renaming it, or appending to the end of the file. In this implementation, I choose to append to the end of the file. This is implemented by opening the file in append mode in the Impl::setupOutputFile function.


  1. void InertialMonitor::Impl::setupOutputFile(const std::string filename) {  
  2.   try {  
  3.     closeOutputFile();  
  4.     outputFileLock.lock();  
  5.     outputFile.open(filename.c_str(), std::ios::app); // append  
  6.     outputFile_p = true;  
  7.     outputFileLock.unlock();  
  8.   } catch (std::exception& e) {  
  9.     qiLogError("InertialMonitor") << "Failed to open the output file: "  
  10.                                   << e.what() << std::endl;  
  11.     throw InertialMonitorError("Failed to open the output file");  
  12.   }  
  13. }  


We also need to ensure that the record file is closed in all cases of execution.

The client inmon_client.cpp


The client program is meant to give us a way to try out our module by calling tis public NAOqi functions, rather than being a very convenient front-end to the inertial sensors. So, it's just a simple command-line program that accepts a few options, conveniently implemented using boost::program_options. This argument-processing code is all in the parseOpt function. This function is quite boring to read, in fact I invite you to have a look at this neat blog post if you want to better understand it. Rather, let's look at the main() function, which contains a few fundamental idiom of NAO programming:


  1. int main(int argc, char* argv[]) {  
  2.   std::string parentBrokerIP, outputFile;  
  3.   int parentBrokerPort, periodInMilliseconds;  
  4.   command_t command; // command_t is just a typedef'd enum with START and STOP.  
  5.   setlocale(LC_NUMERIC, "C"); // Need this to for SOAP serialization of floats to work  
  6.   
  7.   parseOpt(&parentBrokerIP, &parentBrokerPort, &command, &periodInMilliseconds,  
  8.            &outputFile, argc, argv);  
  9.   boost::shared_ptr<AL::ALProxy> inertialMonitorProxy;  
  10.   try {  
  11.     inertialMonitorProxy = boost::shared_ptr<AL::ALProxy>  
  12.         (new AL::ALProxy("InertialMonitor", parentBrokerIP, parentBrokerPort));  
  13.   } catch (const AL::ALError& e) {  
  14.     std::cerr << "Could not create proxy: " << e.what() << std::endl;  
  15.     return 3;  
  16.   }  
  17.   
  18.   if (command == STOP) {  
  19.     inertialMonitorProxy->callVoid("stopMonitor");  
  20.   } else { // (command == START}  
  21.     if (outputFile ==  "") {  
  22.       inertialMonitorProxy->callVoid("startMonitor", periodInMilliseconds);  
  23.     } else {  
  24.       inertialMonitorProxy->callVoid("startMonitorAndRecord", periodInMilliseconds,  
  25.                                      outputFile);  
  26.     }  
  27.   }  
  28.  return 0;  
  29. }  


The line inertialMonitorProxy = boost::shared_ptr<AL::ALProxy>(new AL::ALProxy("InertialMonitor", parentBrokerIP, parentBrokerPort)) is used to get a proxy to the InertialMonitor module without having an InertialMonitorProxy type, which Aldebaran modules have, but not user-created modules. Using the generic-type proxy, we can call the module's API functions using callVoid or, for functions with return values, the templated call<type>, as in inertialMonitorProxy->callVoid("startMonitor", periodInMilliseconds);

Trying it out


Let's try to build the project and call inmon_client.

qibuild configure -c local

qibuild make -c local

Calling it without argument:

./build-local/sdk/bin/inmon_client

outputs:

missing required option command
Allowed options:
  --pip arg (=nao.local) IP of the parent broker. Default: nao.local
  --pport arg (=9559)    Port of the parent broker. Default: 9559
  -c [ --command ] arg   START or STOP
  --output-file arg      filename where to record sensor values on the robot. 
                         Default: no output file

Giving it the required arguments still does not make the module run:

./build-local/sdk/bin/inmon_client --pip 192.168.2.3 -c START

[INFO ] Starting ALNetwork
[INFO ] NAOqi is listening on 127.0.0.1:54010
Could not create proxy: ALNetwork::getModuleByName
failed to get module InertialMonitor http://192.168.2.3:9559
[INFO ] Stopping ALNetwork
[INFO ] Exit

We have to load our module on NAOqi on the robot first.

Loading and autoloading a local module 


First, let's send it over:

qibuild deploy -c atom nao@192.168.2.3:naoqi

If it doesn't fail, we can connect to the robot and look at that ~/naoqi folder

ssh nao@192.168.2.3
cd ~/naoqi; ls

We note the preferences folder. It contains a file called autoload.ini

ls lib/naoqi

In the ~/naoqi/lib/naoqi folder, we have out inmon library (libinmon.so)

If we had produced and deployed other libraries that are not NAOqi modules, they would be directly in ~/naoqi/lib. However, this place is not a conventional place for the shared library loader to look for files. To solve that problem, we can set an environment variable called LD_LIBRARY_PATH to include those paths that are not standard but where we want to put our development libraries. We can do that temporarily by running export LD_LIBRARY_PATH=$HOME/naoqi/lib:$HOME/naoqi/lib/naoqi in our terminal. But it will disappear after we log out of NAO. To solve that, we need to edit ~/.bash_profile and add that same line. As we are on NAO through plain ssh, we have to use a console text editor like emacs or nano: nano ~/.bash_profile (use ctrl-o to write the file, ctrl-x to exit). We can also just drop the line in the file if we're sure that it will not mess it up: echo 'export LD_LIBRARY_PATH=$HOME/naoqi/lib:$HOME/naoqi/lib/naoqi' >> ~/.bash_profile

To ensure that the change is taken into account, let's restart NAO. First, disconnect from ssh (ctrl-d will do that fast.). Then, turn off NAO and on again. Reconnect using ssh.

Then, we need to tell NAOqi to load our new module. Edit ~/naoqi/preferences/autoload.ini. Under the [user] section, add the full path to our module: /home/nao/naoqi/lib/naoqi/libinmon.so

If the file is empty, put the following inside:

[user]
/home/nao/naoqi/lib/naoqi/libinmon.so

[python]

[program]

This file lets NAOqi load what is listed inside when it starts. Under [user], you write /the/full/path/to/your/liblibraryname.so. If you have a Python module, you write /the/full/path/to/your/python_module.py under [python].
The [program] section makes NAOqi autoload an executable program. You need to specify /the/full/path/to/your/program. Lines starting with # are comments.

We manually stop NAOqi (ensure that stiffness is off and NAO is stable):

nao stop

Note that the chest light should now flash yellow, indicating that NAOqi is not running. We restart NAOqi:

nao start

"NAO Gnuk!" - We can examine the NAOqi log:

less /var/log/naoqi/head-naoqi.log

In the end of that file, there should be a line that says "[INFO ] Starting InertialMonitor". If there was a problem and our module crashed, then NAOqi may have crashed too. The log should reflect that, as by default, all logged data at error level is recorded.

But restarting NAO in that way and going to fish for the log file is not very convenient. Moreover, the log only records data logged at INFO level and above. We maybe want to get access to low-level logged data. Let's stop naoqi again:

nao stop

Another way to run naoqi is to call the binary directly. It allows passing naoqi-specific options; for instance, we want to get down to the VERBOSE level of logging:

naoqi -v

You can stop it with ctrl-c.

Usually, I have a few terminal open with ssh into NAO, one of which I use to manipulate naoqi, and the others for files, etc…, and a few more terminals open with local shells.

let's start naoqi at high verbosity and call the startMonitorAndRecord function of the InertialMonitor module instance using the inmon_client binary (it could be run from our computer or from NAO, let's try from out our computer)

in one of the ssh'd terminal:

naoqi -v > ~/naoqi.log 2>&1

in the local shell:

./build-local/sdk/bin/inmon_client --pip 192.168.2.3 --pport 9559 -c START --output-file /home/nao/inmon.out.txt -p 1000

Wait a few seconds, then:

./build-local/sdk/bin/inmon_client --pip 192.168.2.3 --pport 9559 -c STOP

You can stop (ctrl-c) naoqi in the ssh'd terminal, and examine the inmon log file (q to quit):

less ~/inmon.out.txt

and you can have a look at the verbose naoqi.log file that we made too:

less ~/naoqi.log

(search for "Inertial" by typing /Inertial and pressing enter. press n for the next occurrence. Esc to stop. q to quit.)

In the next episode


We have the basis of our module in place. In a second iteration, we will look at:

  • making the module send those sensor values as a stream to another computer on the network (which involves network I/O programming)
  • letting the client program optionally display the received stream of sensor values, and maybe do other neat things with it.



10 comments:

  1. Very nice work! I am halfway through reading

    ReplyDelete
  2. Thanks :D
    I'm not too boring there? Read on a bed, to fall asleep comfortably :P

    ReplyDelete
  3. Hello,

    I get an error when trying to configure. it tells me
    No such target: call_fgrab

    I am using the geode, do you know if this is different than the atom where call_fgrab might not exist?

    ReplyDelete
  4. Hi.

    Remove it from CMakeLists.txt. It's a copy-paste mistake, an old module of mine.

    ReplyDelete
  5. Hello,
    I get a thread error in future.hpp when trying to qibuild make -c local

    How can I fix that?

    Thanks

    ReplyDelete
  6. Hello,

    I get an error in the inmon_client.cpp, It tells me:
    undefined reference to `boost::program_options

    ReplyDelete
  7. Helle. please could youg give the entire code for inmond.cpp

    ReplyDelete
  8. Do you have any idea why i get this: Error was: /home/nao/naoqi/lib/naoqi/libinmon.so: wrong ELF class: ELFCLASS64
    ??

    hope you can help me :)

    ReplyDelete
  9. Any ideas??
    [ERROR] Error was: Launcher::loadLibrary
    Could not load library: "/home/nao/naoqi/lib/naoqi/libinmon.so"

    Error was: libqimessaging.so: cannot open shared object file: No such file or directory

    hope you can help me :D

    ReplyDelete