0

Maze Robot - program design

Zašto ovako?

Any normal Arduino course starts with something very alike hello-world-program in which a program prints "Hello world" or does a similar, simple task. While it is possible for us to go the same way, the process would be very long before we get any useful robot actions. There is different approach, not to use Arduino development environment, but to write a program that accepts only a limited user scripts, like Lego or Fischertechnik. We decided to use a third way, to give You total control and get the results fast. The downside will be that You will immediately start working in a complex, object-oriented, Arduino program. Follow the instructions and nothing bad will happen. Later, we will explain the details of this monster. Just imagine that we started with CAN Bus messaging to sensors and motors... You would be dead-bored before the robot started doing anything.

On the other hand, be ready for a steep learning curve. You will have to grasp rather advanced concepts right in the beginning. That will make sure that You will have much less problems later. The right concept will reveal itself later, cutting development time of any advanced robot manyfold.

If You are more oriented towards smaller code examples, related to specific hardware options, and concept presented here doesn't seem right for You, use this page to see other approach to the problem. You will be able to program CAN Bus and other low-level features.

Program structure

Open header file C:\Users\[Your login]\Documents\Arduino\libraries\mrm-robot\src\mrm-robot-maze.h. The code has rather ample documentation. Here is a general view.

There are 3 main types of classes.

  • Tile. This class defines tiles in the maze. They form a single linked list, thus forming also the maze itself.
  • RobotMaze. As the name suggests, this class represent the robot.
  • Many classes derived from ActionBase base class and they represent various actions.

1. design decision - a linked list

Why a linked list is used? This decision doesn't seem natural at all.

  1. Maze is an obvious candidate for a 3-dimensional C-array and not for a list.
  2. The chosen architecture (microcontroller - MCU) features a quite poor memory management. Linked list will be stored in heap, a part of memory which is dynamically allocated, for example with "new" operator. The manager cannot rearrange the data to fill the holes when objects are deleted.
  3. Speed - traversing a long chain of nodes is surely slower than accessing array's indexed data. Furthermore, the decision is not to store some speed-up data, like end of chain, causing even more traversing. Stack memory area (where array would reside) is also faster than heap.

Of course, it would not be worth the effort to have a poor solution, so some bright spots follow.

  1. Right, maze is a 3D matrix, but not a perfect one. It is sparsely populated. Reserving that space in limited memory is a big waste of resources. Even worse, the robot doesn't know in advance in which direction the maze protracts how much, causing even more space waste for a bigger matrix (C array), or a complicated algorithm for tile reindexing. Still worse, the maze contains checkpoints and it is necessary to remember all the data that exist in the moment robot enters the checkpoint, effectively, depending on implementation, probably doubling the needed space. A linked list is much, much smaller. There is also a quite elegant solution for checkpoints: just prune the part of the list after the checkpoint after returning to it. Zero bytes needed! Even better, You can return the robot to any tile and prune the rest. If the robot missed the checkpoint (did not measure reflective surface correctly), it will still most probably be able to deduce its position. And total memory usage can be maybe 1/10 of the arrays'. We have a clear winner here: linked list.
  2. Memory management is a good argument. It is indeed better to use arrays, but this program uses heap only in fixed-length blocks. In this case, no fragmentation will occur. It even doesn't use strings, that also go into heap and would cause fragmentation.
  3. Speed - there is a golden rule for programming: first measure, then optimize. The result: traversing a linked list is a very fast and causes no visible delay. No optimization needed.

2. decision - inheritance and polymorphism

There are different kinds of robots that can be built using ML-R parts. There can be various programs, one for each robot. This would lead to repeated code - another important thing to avoid by all means. The solution that is used here is C++ inheritance: an object-oriented concept that is beyond scope of this short introduction. Just a short example, that is used id ML-R robotic code, also in this robot. We have a general concept of robot and some more specific robots: RCJ Rescue Maze robot and RCJ Rescue Line robot. This is "a kind of" relationship. Maze robot is "a kind of" robot.

Without further going into details let's see what benefits we can get from this concept. All the general code will be a part of robot's program and we will can this part "Robot class". All the code that is specific for RCJ Rescue Maze will form a part called "RobotMaze class".

Where will a code that a specific sensor go? It may be used by RobotMaze, but some other kinds of robots may use it. We will put it into Robot class. Where will the other code for sensors and effectors (like motors) go? We will put them all into Robot class. Well, most of it. If there is a specific function for some sensor, valid just for RobotMaze, this part will go into RobotMaze class. CAN Bus handling and many other things will also form Robot class.

The best part is that all that Robot class code doesn't need to be visible for the programmer of the RobotMaze and will not clutter its program space. He will only need to change RobotMaze code, like maze algorithm, way the data is stored for tiles, etc. We will have a clear separation of both data and functions, which can be developed and tested independently.

We split the code among different classes, but how we will use it? A general example: let's imagine that You have to develop a program that has to drive a robot along a given trajectory. The robots can have quite different motor groups, for example soccer robots with 4 omni wheels that are positioned in a funny way, tank-like omni-wheel robots, etc. All You know is that they all are derived from the base clase RobotBase and they all implement a common function (go()) for driving the robot along any trajectory needed in Your task. Now comes polymorphism to rescue You. If Your function takes a pointer to RobotBase as an argument, all You have to do is use go() function in Your code. C++ will compile fine and will determine, in runtime, object's (robots) actual derived class and will invoke the correct implementation! So, You will not know which robots will use Your function and how to drive them, but they will still work all fine. If we program some other functions in this way, we will be able to make a program, not knowing which kind of robot will be using it! This is very powerful concept. You cannot do that in C.

3. decision - actions as classes

A robot program loops continually. A way to implement this behaviour is to have a main loop. Therefore, Arduino has one. How to program different actions using Arduino loop? In fact, it cannot be done well. You could have a big "switch" statement in the loop, calling each action's function. However, the different functions must use some common data. Function signatures can become quite long and there will always be some common data that will be implemented as global variables - and that is a bad programming design. Not to mention that this loop will get bigger and bigger: a real monster, without a clear separation of data and actions. So, the decision we made is not to use Arduino loop.

Instead we made a loop internal to RobotMaze class, which uses other classes (derived from ActionBase) as actions.

Actions serve a few purposes.

  • They encapsulate in classes actions robot has to perform. So, we have classes for robot's parts, but here also for non-material terms.
  • No global variables are used. When an information should be shared between one (but called repeatedly) or more functions, it will be stored inside the action object. For example, all the start conditions will be in the object itself.
  • You can use inheritance to indicate relationships between actions, which indeed exist. For example, a movement can be movement straight ahead or turning.
  • You can use in a consistent way actions defined for the base robot, without its code being exposed here.
  • The actions are included in menus just by including a parameter in the constructor call.
  • Buttons can be used to start actions, as well as menu commands. Menus are displayed both in the connected PC and a Bluetooth device, like a mobile phone, and any of the 2 can be used to issue commands.

4. decision - frequency of the main loop

Program flow in a robot's program consists of a loop, that is constantly run and which invokes other functions. The question is: how much time shall these functions consume before returning to the main loop? Should they return almost immediately or there is no such a pressure? In other words, should the functions contain time consuming local loops?

Well, it is definitely easier to allow them to have some. For example, if a function's task is to turn the robot by 90º, it will surely be easier to have a local loop that compares compass to the target value, before returning to the main loop. Obviously, this is profligate local loop, depriving the main loop of its higher frequency. Without that local loop, the main loop will have to call the function many times, each time checking if the target condition is met and this is harder to do. The function will be more complicated as it should do some preparing work in the first run (like setting the target angle value), that mustn't be done in following runs. So, it has to know if it is a first run or not. There are other problems, too.

There are some disadvantages of the described local-loop strategy. The frequency of the main loop is radically decreased, below 1 Hz. That is way to low for some work that has to be done regularly, like exchanging CAN Bus messages, blinking LEDs, or checking for some other events. True, some of there actions can be done using timer interrupts, but this may be awkward and complicates programming logic and debugging, as the program flow is no more deterministic, but rather jumping from one part into another. Interrupts also burden the MCU and can, in extreme cases, choke the program.

In our opinion, it is better to have a higher main loop frequency and we will show this approach. Action classes will mitigate problematic parts as the can contain data which can be used in different parts of program.

5. decision - coordinate systems

There are 2 of them: looking from inside of the robot, our outside (maze's perspective). Therefore, "right" has a relative meaning. To make understanding of the program easier and to offload some mind-bending considerations from the programmer, the program takes all the function's parameters from maze's view. For example, if want the robot to go to the tile that is left of the current tile, we instruct it to go to left. It doesn't matter to which direction it is pointing at the moment. It will obey our instruction and the program will figure out in which direction to rotate. We took advantage of C++ enum structure and fact the its members are in fact natural number, enabling arithmetic operations. Instead of long "if" or "switch" statements, only a verbatim calculations are used.

6. decision - data compression

This program's a ancestor was used in Robocup Junior world championship in distant year 2014, in João Pessoa, Brazil. The robot achieved 3rd place using an Arduino Mega 2560 board. It was challenging to put the labyrinth and checkpoint backup data in 8 MB of RAM. ESP32 doesn't have such a severe memory limitation but we still like to pack bit-like data into bytes. For example a byte can hold 8 boolean variables. Defining 8 boolean variables in C++ will consume 8 bytes of memory. When there is a repetitive structure, like maze's tile, some compression makes sense. This is transparent for programmer because conversion functions are provided.