ECE3411 – Fall 2017 Lec5a.

### Bus and Communication Interfaces Task Based Programming Revisited Real Time Operating Systems

### Marten van Dijk

Department of Electrical & Computer Engineering University of Connecticut Email: marten.van\_dijk@engr.uconn.edu

Copied from Lecture 6a, ECE3411 – Fall 2015, by Marten van Dijk and Syed Kamran Haider

UCONN











8

### MicroWire ( $\mu$ Wire)

- Essentially a subset of SPI
- SPI mode  $0 \rightarrow$  (CPOL, CPHA) = (0, 0)
- Often found in half duplex "three-wire mode"
- Common bi-directional serial data line ightarrow only three wires needed (SIO, SCLK, CS)
- Used in e.g. RTCs (real-time clocks) and serial EEPROMs

### Task Based Programming







| Example Cont'd                                                                                                                                                                                           |    |
|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----|
| Suppose t1=20, m1=1, t2=25, and m2=15                                                                                                                                                                    |    |
| What is the frequency f1 in Hz at which task1() is called?                                                                                                                                               |    |
| What is the frequency f2 in Hz at which task2() is called?                                                                                                                                               |    |
| Answer:                                                                                                                                                                                                  |    |
| <ul> <li>During the time that task2() is executed (which takes 15 ms), task1_timer (which initial value is 20) is decremented to a value v&lt;=5.</li> </ul>                                             |    |
| <ul> <li>The MCU will be idle for v ms after which task2_timer is decremented to 25-15-v and task1_timer just<br/>turned into 0.</li> </ul>                                                              |    |
| <ul> <li>So, after v ms task1() is executed taking 1ms during which task1_timer reduces to 19 and task2_timer<br/>reduces by 1 to 9-v.</li> </ul>                                                        |    |
| <ul> <li>The MCU will be idle for another 9-v ms after which task1_timer is equal to 10+v and task2_timer just<br/>turned into 0.</li> </ul>                                                             |    |
| <ul> <li>Now task2() is executed (which takes 15 ms) after which task1_timer is equal to 0 and task2_timer is equal to 10.</li> </ul>                                                                    |    |
| <ul> <li>The same argument is now repeated for v=0 showing that the execution pattern converges to a repetition of task2() (takes 15 ms), task1() (takes 1 ms), idle time (takes 9 ms) giving</li> </ul> |    |
| • a frequency f_1=f_2=1000/25 Hz.                                                                                                                                                                        | 12 |



- Suppose t1=4, m1=1, t2=8, and m2=4.
- Assume initially task1\_timer = 0 and task2\_timer = t2
- What is the average frequency f1 in Hz at which task1() is called?
- What is the average frequency f2 in Hz at which task2() is called?
- Answer:
  - Task 1 executes during the intervals [12n, 12n+1], [12n+5, 12n+6], for integers  $n \ge 0$ .
  - Task 2 executes during intervals [12n+8,12n+12] for integers n>=0.
  - This gives frequencies f\_1=1000\*2/12 Hz and f\_2=1000/12 Hz.





An organized collection of software extensions of hardware that serve as...

- control routines for operating a computer (for example, to gain access to computer resources (like file I/O))
- an environment for execution of programs



## <section-header><section-header><section-header><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item>











### **Components OS Kernel**

- Task Scheduler: To determine which task will run next in a multitasking system
- Task Dispatcher: To perform necessary bookkeeping to start a task
- Intertask Communication: To support communication between one process (i.e. task) and another

### **Realtime Kernel Design Strategies**

- Polled Loop Systems
- Interrupt Driven Systems
- Multi-Tasking
- Foreground/Background Systems

### Polled Loops

- Simplest RT kernel
- A single and repetitive instruction tests a flag that indicates whether or not an event has occurred
  - \* Examples: Non-blocking LCD instructions, Non-blocking "get string" over the UART channel
- No intertask communication or scheduling needed. Only single tasks exist
- Excellent for handling high-speed data channels, especially when
  - Events occur at widely spaced intervals and
  - Processor is dedicated to handling the data channel

### **Polled Loops**

- Pros:
  - Simple to write and debug
  - Response time easy to determine (as compared to our task-based programming example with two rather than a single task)

### Cons:

- Can fail due to burst of events
- Generally not sufficient to handle complex systems
- · Waste of CPU time, especially when event being polled occurs infrequently

26

### Using Polled Loops

- Often used inside other real time schemes to, e.g.,
  - Poll a suite of sensors for data
  - Check for user inputs (keyboard, keypad, UART data)
- Opposite of interrupt driven systems

## <section-header><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item><list-item>

### ISR

- ISR is a program run in response to an interrupt
  - Disables all interrupts
  - Clears the interrupt flag that got it called
  - Runs code to service the event
  - Re-enables interrupts
  - Exits so the processor can go back to its running task
- Should be as fast as possible, because nothing else can happen when an interrupt is being serviced (when interrupts happen very frequently, tasks are being stalled and progress very slowly, in the worst case one instruction per ISR)
- Interrupts can be
  - Prioritized (service some interrupts before others)
  - Disabed (processor doesn't check or ignores all of them)
  - Masked (processor only sees some interrupts)









| There are four combinations of SCK phase and polarity with respect to serial data, which are determined by control bits CPHA and CPOL. |                       |                  |                  |  |
|----------------------------------------------------------------------------------------------------------------------------------------|-----------------------|------------------|------------------|--|
| Table 18-2. SP                                                                                                                         | I Modes<br>Conditions | Leading Edge     | Trailing eDge    |  |
| 0                                                                                                                                      | CPOL=0, CPHA=0        | Sample (Rising)  | Setup (Falling)  |  |
| 1                                                                                                                                      | CPOL=0, CPHA=1        | Setup (Rising)   | Sample (Falling) |  |
| 2                                                                                                                                      | CPOL=1, CPHA=0        | Sample (Falling) | Setup (Rising)   |  |
| 3                                                                                                                                      | CPOL=1, CPHA=1        | Setup (Falling)  | Sample (Rising)  |  |
|                                                                                                                                        |                       |                  |                  |  |











### Task1: SPI Loopback Testing

Write a simple program to test SPI in loopback mode. In particular:

- Configure SPI in Master mode
- Read a potentiometer's voltage through ADC every 100ms (only upper 8 bits).
- Transmit the byte containing voltage reading over SPI.
- Loopback the transmitted byte by connecting MOSI and MISO pins together according to the instructions given on previous slide.
- Print on LCD the byte value received over SPI.

### Task2: SPI Master Slave Communication

Extend Task1 so that you can exchange voltage readings between yours and your friend's board over SPI bus.

- Configure your board as SPI Master and ask your friend to configure his as SPI Slave.
- Make proper wire connections of SS, SCK, MOSI and MISO pins between the two boards.
- The slave MCU should read its ADC value and write it to SPDR register every 50ms.
- Transmit Master's voltage value every 100ms. This will also result in receiving the last voltage value written in SPDR register in the slave MCU.
- For both Master and Slave, print both transmitted and received values on LCD.

Homework: Use SPI interrupts on both Master and Slave sides for non-blocking SPI communication.





### Polled Loops

- Simplest RT kernel
- A single and repetitive instruction tests a flag that indicates whether or not an event has occurred
  - Examples: Non-blocking LCD instructions, Non-blocking "get string" over the UART channel
- No intertask communication or scheduling needed. Only single tasks exist
- Excellent for handling high-speed data channels, especially when
  - Events occur at widely spaced intervals and
  - Processor is dedicated to handling the data channel

# <section-header><section-header><section-header><list-item><list-item><list-item><list-item>

### Multitasking

- Separate tasks that share one processor (or processors)
- Each task executes within its own context
  - Owns processor
  - Sees its own variables
  - May be interrupted
- Tasks may interact to execute as a whole program



8

## Context Switching

- When the CPU switches from one task to running another, its is said to have switched contexts
- Save the minimum needed to restore the interrupted process
  - Contents of registers
  - Contents of the program counter
  - Contents of coprocessor registers (if applicable)
  - Memory page registers
  - Memory-mapped I/O
  - Special variables
- During context switching, interrupts are often disabled
- Real time systems require minimal times for context switches

## Multitasking

- How do many tasks share the same CPU?
  - Cyclic executive systems
  - Round robin systems
  - Pre-emptive priority systems





















### **Full Featured RTOS**

- Expand foreground/background solution
  - Add network interfaces
  - Add device drivers
  - Add complex debugging tools
- Most common choice for complex systems
- Many commercial operating systems available

### Choosing a RTOS approach

- How do you know which one is right for your application?
- Look at what is driving your system (arrival pattern of data)
  - Irregular (known but varying sequence of intervals between events)
  - Bursty (arbitrary sequence with bound on number of events)
  - Bounded (minimum interarrival interval)
  - Bounded with average rate (unpredictable event times, but cluster around mean)
  - Unbounded (statistical prediction only)
- What is the critical I/O?
- Are there absolute hard deadlines?

### Choosing a RTOS approach

How do you know which one is right for your application? Let's look at some real life choices.

- Reusable Launch Vehicle for satellites. Thrust Vector Control SW requires new attitude data every 40 msec or rocket becomes unstable.
  - · We chose cyclic executive.
- Navigation and Control System for submarine. Interface to multiple sensors at multiple data rates. Information from the Inertial Reference Unit is most critical, but <u>exact timing</u> of input data is not essential.
  - We chose preemptive priority scheme running on a commercial RTOS. Important tasks given highest priority.

### Choosing a RTOS approach

How do you know which one is right for your application? Let's look at some real life choices.

- Avionics System requires new data from flight control surfaces, navigation equipment, and radar system every 50 msec.
  - Cyclic executive. Each task runs to completion. Tasks run in series. Last tasks may not finish before 50msec interrupt occurs.
- Microcontroller running to switch radar antennae and check for incoming signal. If the signal is there, power up the signal processing chip.
  - We chose polled loop.

22







|                                  | DAC SPI Frame Format |            |             |             |              |            |           |
|----------------------------------|----------------------|------------|-------------|-------------|--------------|------------|-----------|
| MCP4921                          | receives a 1         | 6-bit word | from the MC | CU in two 8 | 3-bit SPI tr | ansactions |           |
| below.<br>REGISTER<br>Upper Half |                      | RITE COMM  | AND REGIS   | TER         |              |            |           |
| W-x                              | W-x                  | W-x        | W-0         | W-x         | W-x          | W-x        | W-x       |
| VV-~                             |                      | <u> </u>   |             | D / /       | D10          | D9         | D8        |
| Ā/B                              | BUF                  | GA         | SHDN        | D11         | D10          | 09         |           |
|                                  | BUF                  | GA         | SHDN        | D11         | D10          | 09         |           |
| Ā/B                              |                      | GA         | SHDN        | D11         | 010          | 09         |           |
| Ā/B<br>bit 15                    |                      | GA<br>W-x  | W-x         | W-x         | W-x          | W-x        | U8<br>bit |
| A/B<br>bit 15                    | :                    | 1          |             |             |              |            | bit       |

| DAC Command Bits                                          |                                                                                                                                                              |  |  |
|-----------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|
| The upper 4 bits of the 16 bit word are DAC command bits. |                                                                                                                                                              |  |  |
| The descr                                                 | The description of the 16 bit frame bits is as follows:                                                                                                      |  |  |
| bit 15                                                    | A/B: DAC <sub>A</sub> or DAC <sub>B</sub> Select bit<br>1 = Write to DAC <sub>B</sub><br>0 = Write to DAC <sub>A</sub>                                       |  |  |
| bit 14                                                    | BUF: V <sub>REF</sub> Input Buffer Control bit<br>1 = Buffered<br>0 = Unbuffered                                                                             |  |  |
| bit 13                                                    | <b>GA:</b> Output Gain Select bit<br>1 = 1x (V <sub>OUT</sub> = V <sub>REF</sub> * D/4096)<br>0 = 2x (V <sub>OUT</sub> = 2 * V <sub>REF</sub> * D/4096)      |  |  |
| bit 12                                                    | SHDN:       Output Power Down Control bit         1 =       Output Power Down Control bit         0 =       Output buffer disabled, Output is high impedance |  |  |
| bit 11-0                                                  | <b>D11:D0:</b> DAC Data bits<br>12 bit number "D" which sets the output value. Contains a value between 0 and 4095.                                          |  |  |

| The figure below shows the timing of one SPI transaction (command + data) between the MCU and DAC. |                                                 |  |  |  |
|----------------------------------------------------------------------------------------------------|-------------------------------------------------|--|--|--|
| need to implement the same timing th                                                               | ough SPI interface on ATmega328P.               |  |  |  |
|                                                                                                    |                                                 |  |  |  |
| scк 0 1 2 3 4 5 6 7 8                                                                              | 9 10 11 12 13 14 15 (mode 1,1<br>(mode 0,0      |  |  |  |
| SDI XĀ/B \BUF \GA \SHDN \D11 \D10 \D9 \D8 \D7 \                                                    | 2 data bits<br>D6 X D5 X D4 X D3 X D2 X D1 X D0 |  |  |  |
| LDAC                                                                                               |                                                 |  |  |  |
|                                                                                                    |                                                 |  |  |  |

### Task: Controlling LED Glow

Write a simple program to control the glow of a LED using DAC.

In particular:

- Configure the SPI in Master mode.
- Read a potentiometer's voltage through ADC every 100ms (full 10 bit resolution).
- Normalize the 10-bit ADC reading to a 12-bit digital value for DAC.
- Transmit the 4-bit command and 12-bit data value to DAC over SPI.
- Don't forget to generate a LOW pulse at LDAC pin after transmission.
- Print the ADC's and DAC's readings on LCD.

Homework: Use DAC to generate a 100Hz sine wave with a peak-to-peak amplitude of 5V.
ECE3411 - Fall 2017

 Lecse.

 J<sup>2</sup>C

 RedBot & DC Motor

 Servo Motor Control

 Department of Electrical & Computer Engineering<br/>University of Connecticut<br/>Email: marten.van\_dijk@uconn.edu

 Wides on I2C and Servo Motor Control copied from Lecture 7b,<br/>ECE3411 - Fall 2015, by Marten van Dijk and Syed Kamran Haider<br/>Slides on RedBot adapted from Wikipedia and "RedBot Project"<br/>offered by Sung Yuel Park in Spring 2016











| Titles         AVR Functions         Additional HW         SW Scenario                                                                                 |
|--------------------------------------------------------------------------------------------------------------------------------------------------------|
|                                                                                                                                                        |
| Line Follower ADC, PWM, GPIO,<br>UART IR Sensor, H-bridge Based on IR sensor input,<br>RedBot needs to move<br>along a line (black<br>electrical tape) |











# Connecting the servo

- Typically the servo connectors have 3 wires which should be connected as follows:
  - Red  $\rightarrow$  VCC (+5V)
  - Black  $\rightarrow$  GND (0V)
  - Yellow  $\rightarrow$  PWM signal



+5V GND

ECE3411 - Fall 2017 Lab5c I<sup>2</sup>C: Inter Integrated Circuit Marten van Dijk Department of Electrical & Computer Engineering University of Connecticut Email: marten.van\_dijk@uconn.edu



3

## I<sup>2</sup>C Terminologies

- I<sup>2</sup>C (TWI) protocol allows several devices (up to 128) to be connected.
- Each device is identified by a configurable 7-bit address.
- Each device can communicate with any other device
  - The transmitter address the receiver by its 7-bit address.

#### Table 21-1. TWI Terminology

| Term Description |                                                                                                   |  |  |
|------------------|---------------------------------------------------------------------------------------------------|--|--|
| Master           | The device that initiates and terminates a transmission. The Master also generates the SCL clock. |  |  |
| Slave            | The device addressed by a Master.                                                                 |  |  |
| Transmitter      | The device placing data on the bus.                                                               |  |  |
| Receiver         | The device reading data from the bus.                                                             |  |  |











9



- When the TWI has finished an operation and expects application response, the TWINT Flag is set. The SCL line is pulled low until TWINT is cleared.
- When the TWINT Flag is set, the user must update all TWI Registers with the value relevant for the next TWI bus cycle. As an example, TWDR must be loaded with the value to be transmitted in the next bus cycle.
- After all TWI Register updates and other pending application software tasks have been completed, TWCR is written. When writing TWCR, the TWINT bit should be set.
- Writing a one to TWINT clears the flag. The TWI will then commence executing whatever operation was specified by the TWCR setting.

| uint8_t TWI_Master_Transmit(uint8_t Address, uint8_t Data)                                                               |                                                              |
|--------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------|
| TWCR = (1< <twint) (1<<twen);<="" (1<<twsta)="" th=""  =""><th>// Send START condition</th></twint)>                     | // Send START condition                                      |
| <pre>while (!(TWCR &amp; (1&lt;<twint)));< pre=""></twint)));<></pre>                                                    | // Wait for TWINT Flag set.                                  |
| if ((TWSR & OxF8) != START)<br>ERROR();                                                                                  | // Check value of TWI Status Register.                       |
| TWDR = (Address << 1)   (WRITE);                                                                                         | // Load SLA_W (Slave Address & Write) into TWDR Register.    |
| TWCR = (1 << TWINT)   (1 << TWEN);                                                                                       | // Clear TWINT bit in TWCR to start transmission of address. |
| <pre>while (!(TWCR &amp; (1&lt;<twint)));< pre=""></twint)));<></pre>                                                    | // Wait for TWINT Flag set.                                  |
| <pre>if ((TWSR &amp; 0xF8) != MT_SLA_ACK)</pre>                                                                          | // Check value of TWI Status Register.                       |
| TWDR = Data;                                                                                                             | // Load DATA into TWDR Register.                             |
| TWCR = (1< <twint) (1<<twen);<="" td=""  =""><td>// Clear TWINT bit in TWCR to start transmission of data.</td></twint)> | // Clear TWINT bit in TWCR to start transmission of data.    |
| <pre>while (!(TWCR &amp; (1&lt;<twint)));< pre=""></twint)));<></pre>                                                    | // Wait for TWINT Flag set.                                  |
| <pre>if ((TWSR &amp; 0xF8) != MT_DATA_ACK)</pre>                                                                         | // Check value of TWI Status Register.                       |
| TWCR = (1< <twint) (1<<twen)  (1<<twsto);<="" td=""><td>// Transmit STOP condition.</td></twint) (1<<twen) >             | // Transmit STOP condition.                                  |







Department of Electrical and Computing Engineering

## UNIVERSITY OF CONNECTICUT

# ECE 3411 Microprocessor Application Lab: Fall 2017 Problem Set P5

There are <u>6 questions</u> in this problem set. Answer each question according to the instructions given in at least 3 sentences on own words.

If you find a question ambiguous, be sure to write down any assumptions you make. **Be neat and legible.** If we can't understand your answer, we can't give you credit!

Any form of communication with other students is considered cheating and will merit an F as final grade in the course.

SUBMIT YOUR ANSWERS IN A HARDCOPY FORMAT.

Do not write in the box below

| 1 (x/18) | 2 (x/20) | 3 (x/16) | 4 (x/16) | 5 (x/20) | 6 (x/10) | Total (xx/100) |
|----------|----------|----------|----------|----------|----------|----------------|
|          |          |          |          |          |          |                |
|          |          |          |          |          |          |                |

Name:

**Student ID:** 

**1. [18 points]:**The following code tries to implement a 1kHz PWM signal whose duty cycle is varied in way that it results in 10Hz sawtooth waveform. The MCU clock frequency is 16MHz. List all the bugs that you can identify in this code, and mention how would you fix them.

```
uint16_t step = 80;
uint16_t time_period = 16000;
uint16_t duty_cycle = 0;
void main(void)
{
   /* Configuring Timer 1 for PWM generation */
   OCR1A = time_period-1;
   OCR1B = duty_cycle;
   TCCR1A |= (1<<WGM11) | (1<<WGM10);
                                    // turn on Fast PWM mode
   TCCR1B |= (1<<WGM13) | (1<<WGM12); // turn on Fast PWM mode
                                     // Enable Interrupt
   TIMSK1 \mid = (1 < < 0CIE1B);
   TCCR1B |= (1 < CS10);
                                      // Set pre-scaler @ 1
             // Nothing to do
   while(1);
}
ISR(TIMER0_COMPA_vect)
{
   duty_cycle += step;
   duty_cycle = duty_cycle % (time_period-1);
   OCR1B = duty_cycle;
}
// -----
```

- 2. [20 points]: Answer the following short questions:
- **a.** Why is 128 the maximum number of devices that can be connected together in one I2C network?

**b.** In task 1 of Lab 5c, you were required to establish communication between two MCUs via I2C. What wire connections were needed?

- **c.** How many wires does the SPI protocol use?
- **d.** Building on to part(c), what is each wire used for?

- **3.** [16 points]: Encircle the correct answer for the following:
- **a.** SPI and  $I^2C$  protocols are:
  - (a) Asynchronous
  - (b) Synchronous
  - (c) None of the above
- **b.** SPI and  $I^2C$  protocols:
  - (a) SPI is half duplex whereas  $I^2C$  is full duplex.
  - (b) SPI is full duplex whereas  $I^2C$  is half duplex.
  - (c) Both SPI and  $I^2C$  are full duplex.
  - (d) Both SPI and  $I^2C$  are half duplex.
- **c.** Suppose a SPI Slave wants to send 2 byes of data to a SPI Master. Select the correct statement from the following:
  - (a) Slave initiates SPI communication by itself and sends 2 bytes of data.
  - (b) Slave asks the Master to initiate SPI communication and then sends 2 bytes of data to Master.
  - (c) Master initiates SPI communication by itself and sends 2 bytes of data to Slave.
  - (d) None of the above
- **d.** Which one is correct for transferring a data byte from an  $I^2C$  Master to Slave:
  - (a) Master sends Address, Slave ACKs, Master sends Data, Slave ACKs.
  - (b) Slave sends Address, Master ACKs, Master sends Data, Slave ACKs.
  - (c) Slave requests Address, Slave ACKs, Master sends Data, Master ACKs.
  - (d) None of the above

#### 4. [16 points]:

- **a.** What is context switching? List at least three kinds of data which needs to be saved in a context switch.
- **b.** What is the mechanism (as explained in lecture) behind controlling the angle of a servo motor using one PWM signal control?

**5.** [20 points]: *Earliest Deadline First* (EDF) algorithm schedules the task whose deadline is closest in time.

Complete the following chart according to **non-preemptive** version of *Earliest Deadline First* (EDF) scheduling algorithm. The task specifications are given in the following table.

| Task   | Required CPU Time (ms) | Task Period (ms) |
|--------|------------------------|------------------|
| Task A | 1                      | 6                |
| Task B | 2                      | 5                |
| Task C | 4                      | 10               |

The upward arrows  $(\uparrow)$  in the chart show that the task is ready to be scheduled. Hence each next arrow also represents the deadline of the task needed to be executed in the previous period.





**6. [10 points]:** Can you shortly describe what you have learned and feel confident about using in the future?

## End of Problem Set



Department of Electrical and Computing Engineering

## UNIVERSITY OF CONNECTICUT

# ECE 3411 Microprocessor Application Lab: Fall 2017 Problem Set A5

There are 4 questions in this problem set. Answer each question according to the instructions given in at least 3 sentences on own words.

If you find a question ambiguous, be sure to write down any assumptions you make. **Be neat and legible.** If we can't understand your answer, we can't give you credit!

Any form of communication with other students is considered cheating and will merit an F as final grade in the course.

SUBMIT YOUR ANSWERS IN A HARDCOPY FORMAT.

Do not write in the box below

| 1 (x/20) | 2 (x/30) | 3 (x/30) | 4 (x/20) | Total (xx/100) |
|----------|----------|----------|----------|----------------|
|          |          |          |          |                |
|          |          |          |          |                |

Name:

**Student ID:** 

**1. [20 points]:** Below is a program layout with comments explaining what happens during program execution. Also the meaning of all register initializations is given (there is no need to look into the ATmega328P data sheet).

You should pay attention to the main body which initializes Timer1, and polls its value before an ADC measurement in sleep mode starts and after the execution of an "ADC task" finishes. The difference of the two values is converted to micro seconds and added to a variable busy. The goal for busy is to measure the time during which the MCU is doing "useful" work. The code that is related to busy is highlighted with vertical bars.

After the program layout below, the first subproblem asks you what is truly measured by busy in the program and the second subproblem asks you to explain the code which converts the difference to micro seconds.

```
... we assume a clock frequency of 20MHz
... inclusion of packages
... declaration of global variables
                                     . . .
// ------ //
ISR (TIMER0_COMPA_vect)
{
   /* Update task timer */
   if (taskADC_timer >0 ) {--taskADC_timer;}
}
// ------ //
ISR (ADC_vect)
{
   /* Read a 10-bit conversion */
   AinLow = (int)ADCL;
   Ain = (int)ADCH*256;
   Ain = Ain + AinLow;
}
// ------ //
void taskADC(void)
{
   /* Reset task timer */
   taskADC_timer = 400;
   //Convert Ain into a voltage
   voltage = ((1.0*Ain)/1024.0)*5.0;
   ... Some more computation: sometimes taking more and sometimes taking less time ...
   ... However, no matter how long taskADC() takes, its execution is always <= 200 ms ...
}
```

```
int main(void)
{
    ... initialization variables ...
    // set up timer 1 for 3.2 micro second counter increments
L
    TCCR1B = 3;
                    //set prescalar to divide by 64
    //set up timer 0 such that ISR(TIMER0_COMPA_vect) is called every 1 milli second
    OCR0A = 77;
                          //Set the compare reg to 78 time ticks
    TIMSK0 = (1<<OCIE0A); //Turn on timer 0 cmp match ISR</pre>
    TCCR0B = 4:
                          //Set prescalar to divide by 256
    TCCR0A = (1<<WGM01); //Turn on clear-on-match</pre>
    // initialize the ADC
   ADMUX = 6;
                                         // Select ADC Channel 6
    ADCSRA = (1<<ADEN) | (1<<ADIE) + 7 ; // Enable AD converter, enable its interrupt,
                                         // set prescalar (notice that the ADSC bit is
                                         // not set, so no ADC conversion is started)
    SMCR
          = (1 << SM0);
                                         // Choose ADC sleep mode
    sleep_enable();
    sei();
   while (1)
    {
        if (taskADC_timer == 0)
        {
// Measure timer 1
I
            T1poll_before = TCNT1;
            //Perform an ADC measurement in sleep mode, and execute taskADC
            sleep_cpu();
            taskADC();
T
            //Measure timer 1 again and update busy with the amount of micro seconds that
1
            //have passed: every TCNT1 to TCNT1+1 increment takes 3.2 micro seconds.
            T1poll_after = TCNT1;
            if (T1poll_after > T1poll_before) {
                busy += (T1poll_after - T1poll_before)*3.2;
            } else {
                busy += ( (T1poll_after - T1poll_before) + 65536 ) * 3.2;
            }
        } /* end of if (taskADC_timer == 0) */
    } /* end of while(1) */
} /* end of main() */
```

Initials:

The data sheet (section 9.4) writes for the ADC Noise Reduction Mode that "... the SLEEP instruction makes the MCU enter ADC Noise Reduction mode, stopping the CPU but allowing the ADC, the external interrupts, 2-wire Serial Interface address match, Timer/Counter2 and the Watchdog to continue operating (if enabled) ...". This means that all other hardware modules stop working, in particular, the other timers/counters stop incrementing.

- **A.** (16 points) Answer with "never", "sometimes", or "always", whether the execution times (measured in micro seconds) of the following procedures are added into busy variable. Explain your answers.
  - (a) ISR(TIMER0\_COMPA\_vect)

(b) ISR(ADC\_vect)

(c) sleep\_cpu()

(d) taskADC()

**B.** (4 points) The program assumes that taskADC() always takes  $\leq 200$  ms. Use this assumption to explain why the code

```
if (T1poll_after > T1poll_before) {
    busy += ( T1poll_after - T1poll_before ) * 3.2;
} else {
    busy += ( (T1poll_after - T1poll_before) + 65536 ) * 3.2;
}
```

correctly adds to busy the time in micro seconds that passed between the polling of T1poll\_before and the polling of T1poll\_after.

2. [30 points]: Table 1 shows the characteristics of four tasks that need to be scheduled on the MCU.

"Ready Time" indicates when the corresponding task is ready to execute. In this example, all the tasks are ready and want to execute as soon as the system starts, i.e. at time 0. "Required CPU Time" indicates how many time units are needed for the task to finish.

| Task | Ready Time | <b>Required CPU Time</b> |
|------|------------|--------------------------|
| A    | 0          | 10                       |
| В    | 0          | 1                        |
| С    | 0          | 3                        |
| D    | 0          | 4                        |

The following figure shows an example of First Come First Serve scheduling of these tasks assuming the order A, B, C, and then D.



The following table shows the completion times of the tasks and their corresponding wait times (i.e. while the task is suspended and waiting for the CPU) under First Come First Serve scheduling scheme.

| Task | Ready Time | Required CPU Time | <b>Completion Time</b> | Wait Time |
|------|------------|-------------------|------------------------|-----------|
| A    | 0          | 10                | 10                     | 0         |
| В    | 0          | 1                 | 11                     | 10        |
| C    | 0          | 3                 | 14                     | 11        |
| D    | 0          | 4                 | 18                     | 14        |
|      | Av         | erage             | 13.25                  | 8.75      |

 Table 2: Analysis under First Come First Serve Scheduling

#### A. Round Robin Scheduling: (15 points)

Plot how the tasks A, B, C, and D will be scheduled on the CPU under **Round Robin Scheduling** with a **time slice of 1 time unit**. I.e. assuming an order of A, B, C, and D; the tasks take turns for the CPU and each task gets the CPU for 1 time unit in each turn until the task finishes. Assume that no time is wasted in context switching between the tasks.



Use the plot above to complete the following table.

| Table | <b>3:</b> Analysis under Round | Robin Scheduling |   |
|-------|--------------------------------|------------------|---|
|       |                                |                  | í |

| Task | <b>Ready Time</b> | <b>Required CPU Time</b> | <b>Completion Time</b> | Wait Time |
|------|-------------------|--------------------------|------------------------|-----------|
| A    | 0                 | 10                       |                        |           |
| В    | 0                 | 1                        |                        |           |
| C    | 0                 | 3                        |                        |           |
| D    | 0                 | 4                        |                        |           |
|      | Av                | erage                    |                        |           |

#### **B.** Shortest Remaining Time First Scheduling: (15 points)

Plot how the tasks A, B, C, and D will be scheduled on the CPU under **Shortest Remaining Time First Scheduling**. I.e. whichever task needs the shortest amount of CPU time to finish gets the CPU first. Once this task is finished, another task that needs the smallest CPU time is executed. Assume that no time is wasted in scheduling a new task once a task finishes.



Use the plot above to complete the following table.

| Table 4: | Analysis | under Shortest | Remaining | Time First | Scheduling |
|----------|----------|----------------|-----------|------------|------------|
|          |          |                |           |            |            |

| Task | <b>Ready Time</b> | <b>Required CPU Time</b> | <b>Completion Time</b> | Wait Time |  |
|------|-------------------|--------------------------|------------------------|-----------|--|
| A    | 0                 | 10                       |                        |           |  |
| В    | 0                 | 1                        |                        |           |  |
| C    | 0                 | 3                        |                        |           |  |
| D    | 0                 | 4                        |                        |           |  |
|      | Av                | erage                    |                        |           |  |

**3. [30 points]:** Consider the following while loop in the main code:

```
/* If not already equal to 0, task1_timer and task2_timer
   are decremented every 1 millisecond in a timer ISR */
task1_timer = 0;
                    // Initializing. task1 is ready to execute
task2_timer = 0;
                    // Initializing. task2 is ready to execute
while (1)
{
   if (task1_timer == 0) // if task1 is ready to run.
   {
      task1_timer = t1;
      task1();
                  // task1 takes m1 milliseconds to execute.
   }
   if (task2_timer == 0) // if task2 is ready to run.
   {
      task2_timer = t2;
      task2();
                  // task2 takes m2 milliseconds to execute.
   }
}
```

Suppose if t1=5, m1=1 and t2=20, m2=15 then the corresponding execution pattern is shown below:



Here task1() executes for 1 ms, task2() executes for 15 ms, and the MCU is idle for 4 ms. The same pattern starts getting repeated over and over.

In this example pattern, task2() is executed once per 21 ms (a frequency of 1/(21 ms)) and task1() is executed on average two times per 21 ms (an average frequency of 2/(21 ms)).

Similar to the above mentioned example, answer the following questions.

A. Suppose t1=5, m1=1, t2=10, and m2=15. Draw the execution pattern of the two tasks.



What is the frequency f1 in Hz at which task1() is called? What is the frequency f2 in Hz at which task2() is called?

**B.** Suppose t1=20, m1=1, t2=10, and m2=15. Draw the execution pattern of the two tasks.



What is the frequency f1 in Hz at which task1() is called? What is the average frequency f2 in Hz at which task2() is called? **4. [20 points]:** The robot shown below is a RedBot that you will be using in the last lab. It is a line follower which can follow a black line. It has three line sensors in the front encircled by the dashed line. Write a function to control the movement of this RedBot with the help of the given functions.



The functions you can use are:

- void left\_wheel\_forward(void);
- void left\_wheel\_backward(void);
- void right\_wheel\_forward(void);
- void right\_wheel\_backward(void);

The inputs of your program are the values sampled by three line sensors. For each line sensor, suppose that it reads more than T when it is over the black line. And assume the black line is as thick as a single sensor. Fill the program below.

void redbot\_control(int left\_sensor, int middle\_sensor, int right\_sensor)
{

## End of Problem Set



Department of Electrical and Computing Engineering

## UNIVERSITY OF CONNECTICUT

## ECE 3411 Microprocessor Application Lab: Fall 2017 Independent LAB5

There are 2 independent lab questions in LAB5.

You may not discuss independent labs in any way, shape, or form with anyone else and you are not allowed to lookup solutions from other sources.

Any form of communication with other students or looking up solutions is considered cheating and will merit an F as final grade in the course.

### Name:

**Student ID:** 

1. [Pass/Fail points]: In this task, you need to implement *non-preemptive* version of

Shortest Remaining Time First scheduling for a given set of periodic tasks. The task specifications are given in the following table.

| Task     | Required CPU Time (ms) | Task Period (ms) |
|----------|------------------------|------------------|
| task_0() | 400                    | 1000             |
| task_1() | 200                    | 500              |
| task_2() | 100                    | 600              |

#### Table 1: Task Specifications

You are provided with a C source file named scheduling\_SW.c which provides a framework to run these tasks.

You need to implement the function called SRTF\_scheduler() such that it schedules the tasks w.r.t. *shortest remaining time first* scheduling. Once you have implemented the scheduler, connect a UART terminal to your MCU and see the resulting order of the tasks.

Write the resulting pattern of the tasks in the space given below. Just write the task numbers to show the order in which they are scheduled.

| Table 2: | Scheduler | Output |
|----------|-----------|--------|
|----------|-----------|--------|

| No.    | 1 | 2 | 3  | 4  | 5  | 6  | 7  |
|--------|---|---|----|----|----|----|----|
| Task # |   |   |    |    |    |    |    |
| No.    | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
| Task # |   |   |    |    |    |    |    |

Notice that if your scheduler is not working properly, at least one of the task would miss its deadline and a message will be printed on UART followed by suspension of the program.

It would be easier for you to program/verify your scheduler's behavior by plotting the intended behavior on a paper first. For this purpose, you may use the space provided below.



**2. [Pass/Fail points]:** In this task, you need to implement **Non-blocking SPI**. Write a simple program to test SPI in loopback mode. In particular:

- Configure SPI in Master mode with SPI interrupt enabled.
- Configure Timer1, with compare match interrupt enabled, in CTC mode to overflow after every 100ms.
- In Timer1 ISR, initiate an ADC conversion to read a potentiometers voltage (only upper 8 bits) every 100ms.
- In ADC ISR, once the conversion is complete, initiate a SPI transmission to transmit the byte containing voltage reading over SPI.
- Loopback the transmitted byte by connecting MOSI and MISO pins together.
- Once SPI interrupt triggers, print on LCD the byte value received over SPI.

Implement this system by filling in the gaps in the code layout given below. Notice that busy waiting on SPIF flag after initiating a SPI transmission is **not allowed**.

The following code snippet provides the necessary layout and definitions.

```
#define F_CPU 1600000UL
#include <avr/io.h>
#include <avr/pgmspace.h>
#include <inttypes.h>
#include <avr/interrupt.h>
#include <stdio.h>
#include <string.h>
#include "lcd_lib.h"
// SPI related definitions
#define DDR_SPI
                  DDRB
#define SPI_SS
                  2
#define SPI_MOSI
                  3
#define SPI_MISO
                  4
                  5
#define SPI SCK
// Variables
volatile unsigned int Ain;
volatile uint8_t data_byte;
// LCD Strings
char lcd_buffer[17]; // LCD display buffer
const uint8_t LCD_Master[] PROGMEM = "Master: ";
//-----
```

// All initializations

```
void initialize_all(void)
{
  /* Configure LCD, Timer1, ADC and SPI */
}
//-----
// Timer 1 Compare Match A ISR (TCNT1 = OCR1A)
ISR (TIMER1_COMPA_vect)
{
  /* Write your code here */
}
//-----
// ADC ISR
ISR(ADC_vect)
{
 /* Write your code here */
}
//-----
// SPI ISR
ISR(SPI_STC_vect)
{
  /* Write your code here */
}
//-----
/* Main Function */
int main(void)
{
  initialize_all(); // Initialize everything
              // Enable Global Interrupts
  sei();
         // Nothing to do.
  while(1);
}
//-----
```

# End of LAB5