Intro to Computer Engineering

Assignment 7 - How Far Away Is It?

Click here to access the Canvas page with the repository for this assignment.

========

The idea

The last few assignments have focused on serial communication between the Arduino and a Java program running on your laptop computer. First we set up a way of sending data in each direction. Then we established a protocol to send a variety of meaningful messages one byte at a time. In this assignment we will use the protocol as a way to control the operation of a distance measuring system.

During previous studios, we have investigated the various physical elements of the system: LEDs, a button to debounce, the ultrasonic distance sensor, and DC motors driven by analog outputs. In this assignment we will add one more physical element: a servo motor (output). The system we will build is centered around a distance measurement made by the ultrasonic sensor. We will report the measured distance both to the Java program and to a human, with two different displays: (1) a motorized dial, and (2) an LED bar graph (with controllable brightness).

We will put all these elements together to measure distance and actively control how that distance measurement is displayed.

To implement the distance measuring system and its associated Java program, you will need to use the protocol that we have established to send data in each direction. This assignment will also require delta timing to control the operation of the Arduino’s various incoming and outgoing messages and give it the ability to operate in a non-blocking manner. In addition, the Java program will need to operate in a non-blocking manner (for reading from the keyboard as well as incoming messages from the Arduino).

========

The physical system

You have already built almost all of the physical elements of the measurement system in previous studios. This includes LEDs, buttons, and the ultrasonic distance sensor. Here, we will add one more physical component: a servo motor. Sparkfun has an easy to follow tutorial for this new component.

Servo motor

Here is what the servo motor looks like:

servomotor

Sparkfun’s tutorial on servo motors provides a great introduction to servos. The wiring for the servo follows the color code described below.

Pin Description
White Input PWM Signal
Red Power (5V)
Black Ground (GND)

You use the Servo library (the documentation is here) to program the servo motor.

System construction and operation

The figure below gives an approach to positioning the LEDs (we will use 4 LEDs).

leds

Of course, the ultrasonic sensor needs to be positioned such that it points away from the circuit. Bottom line, put some thought into how you physically construct this system, as space is limited.

The design should periodically measure distance using the ultrasonic sensor at 2 second intervals (use delta timing to enforce this). The raw sensor reading should be sent to Java via a protocol message (independent of display mode) and should be displayed by the Arduino in a variety of ways, depending upon the current display mode. The system should operate over a range of 5 cm to 25 cm (0% to 100% scale). Any ultrasonic display on the Arduino that is 5 cm or smaller can be reported as 0% and any reading 25 cm or greater can be reported as 100%. The Arduino display will have 4 modes, and will also be capable of being turned on and off. When off, no display is active (i.e., they both show 0%). When on, the four modes are: (D) dial only, (L) LED bar graph at full intensity, (H) LED bar graph at half intensity, and (B) both dial and LED bar graph (at full intensity). The initial state (on startup) should be B. A press of the button (be sure to debounce it!) will cycle through the following 4 display modes: mode B, mode D, mode L, mode H, B, D, L, H, etc. If the button is pressed, and the display is off, turn the display on (in addition to cycling to the next display mode). The display mode and on/off status can be controlled from Java by pressing keys: 0 to turn the display off, 1 to turn the display on, D or d to set the display to dial mode (D), L or l to set the display to LED bar graph mode at full intensity (L), H or h to set the display to LED bar graph mode at half intensity, B or b to set the display to both mode (B). Note, for logistical reasons described below, you will need to press the desired key followed by the Enter key.

For the dial display mode, attach a device to the shaft of the servo motor so that it can operate as an indicator. The indicator mostly left (10 degree position) should correspond to 0% scale for distance (or the display is off), and the indicator fully right (170 degree position) should correspond to 100% scale. Note, the servo motor doesn’t operate well at the hard limits of 0 degrees or 180 degrees, so we are limiting the range as 10 to 170 degrees. Clearly, the servo needs to be positioned such that the indicator can swing 180 degrees without obstruction.

With 4 LEDs in the bar graph, we can display 5 possible distances. If the display is off, no LEDs should be on. When the LED display is active (L, H, or B display mode), distance readings that are below 20% of scale should have all LEDs off, between 20% and 40% of scale should have one LED on, between 40% and 60% of scale should have two LEDs on, between 60% and 80% of scale should have three LEDs on, and above 80% of scale should have all LEDs on. You should consider the bottom 5 cm that are below a valid reading as being off-scale when establishing thresholds.

The LED display has one additional feature that you will need to implement. The brightness of the LEDs that are on should depend on the display mode, full intensity if in mode L or B, and half intensity if in mode H. Controlling the intensity of an LED is accomplished by using a pulse-width modulated output, the documentation for which is here. It is not required for half intensity to be precisely 50% of full intensity, it is sufficient for the LEDs to be visible but noticably dimmer when at half intensity.

The PC running Java should operate as follows. Any valid protocol message received from the Arduino should be printed (in a reasonable, readable form) to the console. This can mimic the printing from Module 6 for all of the protocol keys that are common to the two assignments. Ultrasonic sensor reading messages should print both the raw value and the converted distance in cm. Keys pressed should have the following effects: 0 should turn off the Arduino display, while 1 should turn the Arduino display on; D or d should command the display to dial mode; L or l should command the display to full intensity LED mode; H or h should command the display to half intensity LED mode; and B or b should command the display to both mode with LEDs at full intensity. The communication of these commands should be communicated to the Arduino through the use of the protocol, using the newly established protocol keys listed below.

We will use the protocol from Module 6 to communication back and forth between the Arduino and Java. For this assignment, we define the following protocol keys (from Arduino to Java):

This assignment also uses the protocol in the other direction. We will also use these protocol keys (from Java to Arduino):

Notice the choice of ASCII character codes for the messages for protocol key 0x41: 0x3d44 is =D, 0x3d4c is =L, 0x3d48 is =H, and 0x3d42 is =B. This was not an accident.

Before you get started on the actual assignment, it is a very good idea to check out the various components that you will use. Using software from the studios or sketches you get from the Sparkfun tutorials (or write on your own), put the ultrasonic distance sensor, the button, the LEDs, and the new servo motor through their paces, making sure they each individually operate as intended. Ensure that the LEDs used for the bar graph are connected to PWM-capable outputs and function at both full intensity and half intensity (note that this is a new thing we are doing with the LEDs).

========

The Arduino assignment

Starting from the DistanceDisplay.ino sketch in your repository (which is mostly empty), structure your loop() code so that it is non-blocking and each iteration checks to see if: (1) the button has been pressed, (2) there is a new byte ready to be read at the serial input cable from the PC, and (3) whether a delta-time timing event is ready.

As we did in studio, when you’re not actively using the ultrasonic sensor, please unplug the Arduino. This helps eliminate interference with other groups.

  1. You will need to define five display states, embodying on/off and four display modes: dial, full intensity LED, half intensity LED or both dial and LED.

  2. At the beginning of your loop you will need to read the state of the button (with debouncing, using delta-timing!) to determine whether it has been pressed. If it has, you need to toggle a state change and change the behavior of the system.

  3. As part of your loop() function you should regularly check to see if a byte has been received on the communications link. Bytes received should be the input to a FSM associated with Java communications that is receiving keys 0x40 and 0x41 of the protocol. When a complete message has been received, toggle the appropriate state change to alter the behavior of the system. Once you have successfully parsed a protocol message, you may optionally send an INFO message back to Java acknowledging the received command.

  4. The loop() function should also have a delta-timing test so that the ultrasonic sensor is read every 2 seconds.

  5. Once every two seconds, send the newest (raw) distance sensor reading to the Java program in a message using the protocol from the previous module. You should do this regardless of which display mode the system is in.

  6. Blocking behavior is OK while reading the ultrasonic sensor, but all other operations should be non-blocking. This includes debouncing the button, reading bytes from the serial communications channel, and implementing the 2 second delay between sensor readings.

  7. Note that the Arduino program might have as many as 3 finite-state machines: one associated with debouncing the button, one associated with recognizing protocol messages, and one keeping track of the display states. You are not required to use FSMs in all these places, but it can make your design considerably easier to reason about, implement, and debug.

========

The Java Assignment

The Java portion of this assignment will regularly receive data from the Arduino in the form of messages that adhere to the protocol from Module 6. I suggest starting with a copy of the Java code (including SerialComm) you used in Module 6 for reading in messages. Be sure to print out any messages you receive. This includes distance readings (which Java will need to convert to cm) and INFO/ERROR messages (i.e., support all of the messages other than the potentiometer reading, which is not required).

Remember that SerialComm’s readByte() method will block until a byte is consumed from the serial port, so never first call it without ensuring that there is a byte to be read. Java needs to be able to both receive incoming messages from Arduino and read input typed by the user that is meant to control the display on the Arduino. Both must be fully non-blocking!

To read input from the console in a non-blocking way, use System.in.available() > 0 to check for bytes. Note, input is line-buffered, so input data is not transmitted until the Enter key is pressed. Then System.in.read() will get one byte from the line buffered by the Enter key. A valid byte read in from the console will be one of the letters that indicate an on/off command or a mode change command. Based on the input byte, construct a message of the appropriate kind using the protocol and send it to the Arduino.

========

The check-in

  1. Commit all your code.
  2. Check out with a TA.

The rubric

Note, this is merely advisory, the precise rubric is on Canvas.

Generated at 2025-03-26 21:50:15 +0000.
Page written by Roger Chamberlain.