Page 209
    kill(pid[i]);
  return 0;
}
void exec_helper(int (∗code_start) (int,char∗∗)) {
  pid[task_index++] = execi(code_start, 0, NULL, 0, DEFAULT_STACK_SIZE);
}
int main() {
  task_index = 0;
  exec_helper(&avoid);
  exec_helper(&seek_enlightenment);
  exec_helper(&cruise);
  exec_helper(&arbitrate);
  execi(&stop_task, 0, NULL, 0, DEFAULT_STACK_SIZE);
  tm_start();
  return 0;
}
LightSeeker.c is a relatively large program, but it consists of easily understandable pieces. As before, I'll start at the bottom and work backwards through the source code.
The main() function simply serves to start the other tasks in the program. A helper function, exec_helper(), is used to start the three behavior tasks, avoid(), seek_enlightenment
(), and cruise(). exec_helper() is also used to start the arbitrate() task, which examines the output of the three behaviors and sends the appropriate command to the motors. The 
exec_helper() function simply starts each task using execi() and stores the returned process ID in an array. Back in main(), stop_task() is also started. When the Run button is 
pressed, stop_task() simply goes through the process ID array that exec_helper() built and kills each process.
arbitrate() examines the output commands of each behavior. If the command is not COMMAND_NONE, the current motor command is set from the behavior. The later behaviors, of course, 
will overwrite the motor command. The last behavior, avoid(), is at the highest level. If it chooses to control the robot, Seek_enlightenment() and cruise() have nothing to say about 
it.
To make it clearer what's going on while the robot is running, arbitrate() writes a character to the display that indicates which behavior is currently active. A "c" on the right side of the 
display indicates that cruise() has control, an "s" stands for seek_enlightenment(), and an "a" shows that the avoid() behavior is controlling the robot.
Page 210
When arbitrate() has determined the motor command, it uses motor_control() to interpret the command and to actually set the direction and speed of the outputs. This design is very 
similar to the design of the NQC RoboTag program, from Chapter 9.
The cruise() behavior is simple; it just sets its command variable to COMMAND_FORWARD ad infinitum.
The next behavior, seek_enlightenment(), is not so simple. The basic idea, however, goes like this:
if I'm seeing darker stuff than I've just been seeing
  look to either side for something brighter
seek_enlightenment() implements this with the idea of a baseline. The initial baseline is taken straight from the light sensor reading:
baseline = process_light(SENSOR_2);