Dalton Altstaetter & Ken Lee

Due 2/13/15

445M

**Lab Report 1**

**Objectives:**

The objectives of Lab 1 are to familiarize ourselves with basic interfacing of the TM4C using the provided memory mapped driver libraries, getting familiar with the Keil environment, and to review the software modules that will be used and built upon this semester(ADC, UART, FIFO, Timers, SysTick, PLL). We also interfaced with an external LCD, ST7735, for debugging and displaying data/information. We split the LCD into two separate displays, which can be used by separate threads. We built a simple interpreter and added commands for debugging the ADC, LCD, and Timer drivers. We also got re-acquainted with the Digital Logic Analyzer when testing our system and gathering profiling information to test the accuracy of our timer interrupts. In addition, we learned basic commands on how to better maintain and integrate our code for future labs (using Git).

**Hardware Design: none**

**Software Design:**

void ST7735\_Message (int device, int line, char \*string, long value){

if(device==0){

if(line>7){

ST7735\_SetCursor(0,0);

ST7735\_OutString((uint8\_t\*)" ");

ST7735\_SetCursor(0,0);

ST7735\_OutString((uint8\_t\*)"line out of bounds");

}else{

ST7735\_SetCursor(0,line);

ST7735\_OutString((uint8\_t\*)" ");

ST7735\_SetCursor(0,line);

ST7735\_OutString((uint8\_t\*)string);

ST7735\_OutChar(' ');

ST7735\_OutUDec(value);

}

}else if(device==1){

if(line<8){

ST7735\_SetCursor(0,8);

ST7735\_OutString((uint8\_t\*)" ");

ST7735\_SetCursor(0,8);

ST7735\_OutString((uint8\_t\*)"line out of bounds");

}else{

ST7735\_SetCursor(0, line);

ST7735\_OutString((uint8\_t\*)" ");

ST7735\_SetCursor(0, line);

ST7735\_OutString((uint8\_t\*)string);

ST7735\_OutChar(' ');

ST7735\_OutUDec(value);

}

}else{

ST7735\_SetCursor(0,0);

ST7735\_OutString((uint8\_t\*)" ");

ST7735\_SetCursor(0,0);

ST7735\_OutString((uint8\_t\*)"Invalid Device");

}

}

// SS3 interrupts: enabled and promoted to controller

volatile uint32\_t NumberOfSamples=0;

volatile uint16\_t\* Buffer;

volatile uint32\_t Status=1;

void ADC\_Collect(uint8\_t channelNum, uint32\_t fs, uint16\_t buffer[],uint32\_t numberOfSamples){

volatile uint32\_t delay;

// \*\*\*\* GPIO pin initialization \*\*\*\*

switch(channelNum){ // 1) activate clock

case 0:

case 1:

case 2:

case 3:

case 8:

case 9: // these are on GPIO\_PORTE

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGCGPIO\_R4; break;

case 4:

case 5:

case 6:

case 7: // these are on GPIO\_PORTD

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGCGPIO\_R3; break;

case 10:

case 11: // these are on GPIO\_PORTB

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGCGPIO\_R1; break;

default: return; // 0 to 11 are valid channels on the LM4F120

}

delay = SYSCTL\_RCGCGPIO\_R; // 2) allow time for clock to stabilize

delay = SYSCTL\_RCGCGPIO\_R;

switch(channelNum){

case 0: // Ain0 is on PE3

GPIO\_PORTE\_DIR\_R &= ~0x08; // 3.0) make PE3 input

GPIO\_PORTE\_AFSEL\_R |= 0x08; // 4.0) enable alternate function on PE3

GPIO\_PORTE\_DEN\_R &= ~0x08; // 5.0) disable digital I/O on PE3

GPIO\_PORTE\_AMSEL\_R |= 0x08; // 6.0) enable analog functionality on PE3

break;

case 1: // Ain1 is on PE2

GPIO\_PORTE\_DIR\_R &= ~0x04; // 3.1) make PE2 input

GPIO\_PORTE\_AFSEL\_R |= 0x04; // 4.1) enable alternate function on PE2

GPIO\_PORTE\_DEN\_R &= ~0x04; // 5.1) disable digital I/O on PE2

GPIO\_PORTE\_AMSEL\_R |= 0x04; // 6.1) enable analog functionality on PE2

break;

case 2: // Ain2 is on PE1

GPIO\_PORTE\_DIR\_R &= ~0x02; // 3.2) make PE1 input

GPIO\_PORTE\_AFSEL\_R |= 0x02; // 4.2) enable alternate function on PE1

GPIO\_PORTE\_DEN\_R &= ~0x02; // 5.2) disable digital I/O on PE1

GPIO\_PORTE\_AMSEL\_R |= 0x02; // 6.2) enable analog functionality on PE1

break;

case 3: // Ain3 is on PE0

GPIO\_PORTE\_DIR\_R &= ~0x01; // 3.3) make PE0 input

GPIO\_PORTE\_AFSEL\_R |= 0x01; // 4.3) enable alternate function on PE0

GPIO\_PORTE\_DEN\_R &= ~0x01; // 5.3) disable digital I/O on PE0

GPIO\_PORTE\_AMSEL\_R |= 0x01; // 6.3) enable analog functionality on PE0

break;

case 4: // Ain4 is on PD3

GPIO\_PORTD\_DIR\_R &= ~0x08; // 3.4) make PD3 input

GPIO\_PORTD\_AFSEL\_R |= 0x08; // 4.4) enable alternate function on PD3

GPIO\_PORTD\_DEN\_R &= ~0x08; // 5.4) disable digital I/O on PD3

GPIO\_PORTD\_AMSEL\_R |= 0x08; // 6.4) enable analog functionality on PD3

break;

case 5: // Ain5 is on PD2

GPIO\_PORTD\_DIR\_R &= ~0x04; // 3.5) make PD2 input

GPIO\_PORTD\_AFSEL\_R |= 0x04; // 4.5) enable alternate function on PD2

GPIO\_PORTD\_DEN\_R &= ~0x04; // 5.5) disable digital I/O on PD2

GPIO\_PORTD\_AMSEL\_R |= 0x04; // 6.5) enable analog functionality on PD2

break;

case 6: // Ain6 is on PD1

GPIO\_PORTD\_DIR\_R &= ~0x02; // 3.6) make PD1 input

GPIO\_PORTD\_AFSEL\_R |= 0x02; // 4.6) enable alternate function on PD1

GPIO\_PORTD\_DEN\_R &= ~0x02; // 5.6) disable digital I/O on PD1

GPIO\_PORTD\_AMSEL\_R |= 0x02; // 6.6) enable analog functionality on PD1

break;

case 7: // Ain7 is on PD0

GPIO\_PORTD\_DIR\_R &= ~0x01; // 3.7) make PD0 input

GPIO\_PORTD\_AFSEL\_R |= 0x01; // 4.7) enable alternate function on PD0

GPIO\_PORTD\_DEN\_R &= ~0x01; // 5.7) disable digital I/O on PD0

GPIO\_PORTD\_AMSEL\_R |= 0x01; // 6.7) enable analog functionality on PD0

break;

case 8: // Ain8 is on PE5

GPIO\_PORTE\_DIR\_R &= ~0x20; // 3.8) make PE5 input

GPIO\_PORTE\_AFSEL\_R |= 0x20; // 4.8) enable alternate function on PE5

GPIO\_PORTE\_DEN\_R &= ~0x20; // 5.8) disable digital I/O on PE5

GPIO\_PORTE\_AMSEL\_R |= 0x20; // 6.8) enable analog functionality on PE5

break;

case 9: // Ain9 is on PE4

GPIO\_PORTE\_DIR\_R &= ~0x10; // 3.9) make PE4 input

GPIO\_PORTE\_AFSEL\_R |= 0x10; // 4.9) enable alternate function on PE4

GPIO\_PORTE\_DEN\_R &= ~0x10; // 5.9) disable digital I/O on PE4

GPIO\_PORTE\_AMSEL\_R |= 0x10; // 6.9) enable analog functionality on PE4

break;

case 10: // Ain10 is on PB4

GPIO\_PORTB\_DIR\_R &= ~0x10; // 3.10) make PB4 input

GPIO\_PORTB\_AFSEL\_R |= 0x10; // 4.10) enable alternate function on PB4

GPIO\_PORTB\_DEN\_R &= ~0x10; // 5.10) disable digital I/O on PB4

GPIO\_PORTB\_AMSEL\_R |= 0x10; // 6.10) enable analog functionality on PB4

break;

case 11: // Ain11 is on PB5

GPIO\_PORTB\_DIR\_R &= ~0x20; // 3.11) make PB5 input

GPIO\_PORTB\_AFSEL\_R |= 0x20; // 4.11) enable alternate function on PB5

GPIO\_PORTB\_DEN\_R &= ~0x20; // 5.11) disable digital I/O on PB5

GPIO\_PORTB\_AMSEL\_R |= 0x20; // 6.11) enable analog functionality on PB5

break;

}

DisableInterrupts();

SYSCTL\_RCGCADC\_R |= 0x01; // activate ADC0

SYSCTL\_RCGCTIMER\_R |= 0x01; // activate timer0

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER0\_CTL\_R = 0x00000000; // disable timer0A during setup

TIMER0\_CTL\_R |= 0x00000020; // enable timer0A trigger to ADC

TIMER0\_CFG\_R = 0; // configure for 32-bit timer mode

TIMER0\_TAMR\_R = 0x00000002; // configure for periodic mode, default down-count settings

TIMER0\_TAPR\_R = 0; // prescale value for trigger

TIMER0\_TAILR\_R = 80000000/fs-1; // start value for trigger assuming 80 MHz

TIMER0\_IMR\_R = 0x00000000; // disable all interrupts

TIMER0\_CTL\_R |= 0x00000001; // enable timer0A 32-b, periodic, no interrupts

ADC0\_PC\_R = 0x01; // configure for 125K samples/sec

ADC0\_SSPRI\_R = 0x3210; // sequencer 0 is highest, sequencer 3 is lowest

ADC0\_ACTSS\_R &= ~0x08; // disable sample sequencer 3

ADC0\_EMUX\_R = (ADC0\_EMUX\_R&0xFFFF0FFF)+0x5000; // timer trigger event

ADC0\_SSMUX3\_R = channelNum;

ADC0\_SSCTL3\_R = 0x06; // set flag and end

ADC0\_IM\_R |= 0x08; // enable SS3 interrupts

ADC0\_ACTSS\_R |= 0x08; // enable sample sequencer 3

NVIC\_PRI4\_R = (NVIC\_PRI4\_R&0xFFFF00FF)|0x00004000; //priority 2

NVIC\_EN0\_R = 1<<17; // enable interrupt 17 in NVIC

NumberOfSamples = numberOfSamples;

Buffer = buffer;

EnableInterrupts();

}

volatile uint32\_t ADCvalue;

void ADC0Seq3\_Handler(void){

static int i=0;

long sr;

ADC0\_ISC\_R = 0x08; // acknowledge ADC sequence 3 completion

sr = StartCritical();

Buffer[i] = ADC0\_SSFIFO3\_R; // 12-bit result

if(i==NumberOfSamples){

Status=1; //ADC conversion complete

ADC0\_IM\_R &= ~0x08; // disable SS3 interrupts when buffer is full

}else{

Status=0; //ADC still filling buffer

i++;

}

EndCritical(sr);

}

int ADC\_Status(void){

return Status;

}

// This initialization function sets up the ADC according to the

// following parameters. Any parameters not explicitly listed

// below are not modified:

// Max sample rate: <=125,000 samples/second

// Sequencer 0 priority: 1st (highest)

// Sequencer 1 priority: 2nd

// Sequencer 2 priority: 3rd

// Sequencer 3 priority: 4th (lowest)

// SS3 triggering event: software trigger

// SS3 1st sample source: programmable using variable 'channelNum' [0:11]

// SS3 interrupts: enabled but not promoted to controller

void ADC\_Open(uint32\_t channelNum){

volatile uint32\_t delay;

switch(channelNum){ // 1) activate clock

case 0:

case 1:

case 2:

case 3:

case 8:

case 9: // these are on GPIO\_PORTE

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGCGPIO\_R4; break;

case 4:

case 5:

case 6:

case 7: // these are on GPIO\_PORTD

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGCGPIO\_R3; break;

case 10:

case 11: // these are on GPIO\_PORTB

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGCGPIO\_R1; break;

default: return; // 0 to 11 are valid channels on the LM4F120

}

delay = SYSCTL\_RCGCGPIO\_R; // 2) allow time for clock to stabilize

delay = SYSCTL\_RCGCGPIO\_R;

switch(channelNum){

case 0: // Ain0 is on PE3

GPIO\_PORTE\_DIR\_R &= ~0x08; // 3.0) make PE3 input

GPIO\_PORTE\_AFSEL\_R |= 0x08; // 4.0) enable alternate function on PE3

GPIO\_PORTE\_DEN\_R &= ~0x08; // 5.0) disable digital I/O on PE3

GPIO\_PORTE\_AMSEL\_R |= 0x08; // 6.0) enable analog functionality on PE3

break;

case 1: // Ain1 is on PE2

GPIO\_PORTE\_DIR\_R &= ~0x04; // 3.1) make PE2 input

GPIO\_PORTE\_AFSEL\_R |= 0x04; // 4.1) enable alternate function on PE2

GPIO\_PORTE\_DEN\_R &= ~0x04; // 5.1) disable digital I/O on PE2

GPIO\_PORTE\_AMSEL\_R |= 0x04; // 6.1) enable analog functionality on PE2

break;

case 2: // Ain2 is on PE1

GPIO\_PORTE\_DIR\_R &= ~0x02; // 3.2) make PE1 input

GPIO\_PORTE\_AFSEL\_R |= 0x02; // 4.2) enable alternate function on PE1

GPIO\_PORTE\_DEN\_R &= ~0x02; // 5.2) disable digital I/O on PE1

GPIO\_PORTE\_AMSEL\_R |= 0x02; // 6.2) enable analog functionality on PE1

break;

case 3: // Ain3 is on PE0

GPIO\_PORTE\_DIR\_R &= ~0x01; // 3.3) make PE0 input

GPIO\_PORTE\_AFSEL\_R |= 0x01; // 4.3) enable alternate function on PE0

GPIO\_PORTE\_DEN\_R &= ~0x01; // 5.3) disable digital I/O on PE0

GPIO\_PORTE\_AMSEL\_R |= 0x01; // 6.3) enable analog functionality on PE0

break;

case 4: // Ain4 is on PD3

GPIO\_PORTD\_DIR\_R &= ~0x08; // 3.4) make PD3 input

GPIO\_PORTD\_AFSEL\_R |= 0x08; // 4.4) enable alternate function on PD3

GPIO\_PORTD\_DEN\_R &= ~0x08; // 5.4) disable digital I/O on PD3

GPIO\_PORTD\_AMSEL\_R |= 0x08; // 6.4) enable analog functionality on PD3

break;

case 5: // Ain5 is on PD2

GPIO\_PORTD\_DIR\_R &= ~0x04; // 3.5) make PD2 input

GPIO\_PORTD\_AFSEL\_R |= 0x04; // 4.5) enable alternate function on PD2

GPIO\_PORTD\_DEN\_R &= ~0x04; // 5.5) disable digital I/O on PD2

GPIO\_PORTD\_AMSEL\_R |= 0x04; // 6.5) enable analog functionality on PD2

break;

case 6: // Ain6 is on PD1

GPIO\_PORTD\_DIR\_R &= ~0x02; // 3.6) make PD1 input

GPIO\_PORTD\_AFSEL\_R |= 0x02; // 4.6) enable alternate function on PD1

GPIO\_PORTD\_DEN\_R &= ~0x02; // 5.6) disable digital I/O on PD1

GPIO\_PORTD\_AMSEL\_R |= 0x02; // 6.6) enable analog functionality on PD1

break;

case 7: // Ain7 is on PD0

GPIO\_PORTD\_DIR\_R &= ~0x01; // 3.7) make PD0 input

GPIO\_PORTD\_AFSEL\_R |= 0x01; // 4.7) enable alternate function on PD0

GPIO\_PORTD\_DEN\_R &= ~0x01; // 5.7) disable digital I/O on PD0

GPIO\_PORTD\_AMSEL\_R |= 0x01; // 6.7) enable analog functionality on PD0

break;

case 8: // Ain8 is on PE5

GPIO\_PORTE\_DIR\_R &= ~0x20; // 3.8) make PE5 input

GPIO\_PORTE\_AFSEL\_R |= 0x20; // 4.8) enable alternate function on PE5

GPIO\_PORTE\_DEN\_R &= ~0x20; // 5.8) disable digital I/O on PE5

GPIO\_PORTE\_AMSEL\_R |= 0x20; // 6.8) enable analog functionality on PE5

break;

case 9: // Ain9 is on PE4

GPIO\_PORTE\_DIR\_R &= ~0x10; // 3.9) make PE4 input

GPIO\_PORTE\_AFSEL\_R |= 0x10; // 4.9) enable alternate function on PE4

GPIO\_PORTE\_DEN\_R &= ~0x10; // 5.9) disable digital I/O on PE4

GPIO\_PORTE\_AMSEL\_R |= 0x10; // 6.9) enable analog functionality on PE4

break;

case 10: // Ain10 is on PB4

GPIO\_PORTB\_DIR\_R &= ~0x10; // 3.10) make PB4 input

GPIO\_PORTB\_AFSEL\_R |= 0x10; // 4.10) enable alternate function on PB4

GPIO\_PORTB\_DEN\_R &= ~0x10; // 5.10) disable digital I/O on PB4

GPIO\_PORTB\_AMSEL\_R |= 0x10; // 6.10) enable analog functionality on PB4

break;

case 11: // Ain11 is on PB5

GPIO\_PORTB\_DIR\_R &= ~0x20; // 3.11) make PB5 input

GPIO\_PORTB\_AFSEL\_R |= 0x20; // 4.11) enable alternate function on PB5

GPIO\_PORTB\_DEN\_R &= ~0x20; // 5.11) disable digital I/O on PB5

GPIO\_PORTB\_AMSEL\_R |= 0x20; // 6.11) enable analog functionality on PB5

break;

}

SYSCTL\_RCGC0\_R |= 0x00010000; // 7) activate ADC0 (legacy code)

// SYSCTL\_RCGCADC\_R |= 0x00000001; // 7) activate ADC0 (actually doesn't work)

delay = SYSCTL\_RCGC0\_R; // 8) allow time for clock to stabilize

delay = SYSCTL\_RCGC0\_R;

// SYSCTL\_RCGC0\_R &= ~0x00000300; // 9) configure for 125K (legacy code)

ADC0\_PC\_R &= ~0xF; // 9) clear max sample rate field

ADC0\_PC\_R |= 0x1; // configure for 125K samples/sec

ADC0\_SSPRI\_R = 0x3210; // 10) Sequencer 3 is lowest priority

ADC0\_ACTSS\_R &= ~0x0008; // 11) disable sample sequencer 3

ADC0\_EMUX\_R &= ~0xF000; // 12) seq3 is software trigger

ADC0\_SSMUX3\_R &= ~0x000F; // 13) clear SS3 field

ADC0\_SSMUX3\_R += channelNum; // set channel

ADC0\_SSCTL3\_R = 0x0006; // 14) no TS0 D0, yes IE0 END0

ADC0\_IM\_R &= ~0x0008; // 15) disable SS3 interrupts

ADC0\_ACTSS\_R |= 0x0008; // 16) enable sample sequencer 3

}

//------------ADC\_In------------

// Busy-wait Analog to digital conversion

// Input: none

// Output: 12-bit result of ADC conversion

uint16\_t ADC\_In(void){

uint32\_t result;

ADC0\_PSSI\_R = 0x0008; // 1) initiate SS3

while((ADC0\_RIS\_R&0x08)==0){}; // 2) wait for conversion done

// if you have an A0-A3 revision number, you need to add an 8 usec wait here

result = ADC0\_SSFIFO3\_R&0xFFF; // 3) read result

return result;

}

uint16\_t TestBuffer[64];

int main(void){

char input\_str[30];

int input\_num,i,device,line;

int freq, numSamples;

PLL\_Init();

UART\_Init(); // initialize UART

Output\_Init(); // initialize LCD

GPIO\_PortF\_Init(); // initialize PortF

OutCRLF();

OutCRLF();

OS\_AddPeriodicThread(&PF1\_Toggle, 1, 100, 4); //Toggle PF1 at 10 Hz

OS\_LaunchThread(&PF1\_Toggle, 1);

//Print Interpreter Menu

printf("Debugging Interpreter Lab 1\n\r");

printf("Commands:\n\r");

printf("LCD\n\r");

printf("ADC\_Open - must call before ADC\_In\n\r");

printf("ADC\_In\n\r");

printf("ADC\_Collect\n\r");

printf("ADC\_Status\n\r");

printf("OS-RTP - OS\_ReadTimerPeriod\n\r");

printf("OS-RTV - OS\_ReadTimerValue\n\r");

printf("OS-CPT - OS\_ClearPeriodicTime\n\r");

printf("OS-ST - OS\_StopThread\n\r");

while(1){

printf("\n\rEnter a command:\n\r");

for(i=0;input\_str[i]!=0;i++){input\_str[i]=0;} //Flush the input\_str

UART\_InString(input\_str,30);

if(!strcmp(input\_str,"LCD")){

printf("\n\rMessage to Print: ");

for(i=0;input\_str[i]!=0;i++){input\_str[i]=0;} //Flush the input\_str

UART\_InString(input\_str,30);

printf("\n\rNumber to Print: ");

input\_num=UART\_InUDec();

printf("\n\rDevice to Print to: ");

device = UART\_InUDec();

printf("\n\rLine to Print to: ");

line = UART\_InUDec();

ST7735\_Message(device,line,input\_str,input\_num);

}

else if(!strcmp(input\_str,"ADC\_Open")){

printf("\n\rChannel to Open: ");

input\_num=UART\_InUDec();

ADC\_Open(input\_num);

}

else if(!strcmp(input\_str,"ADC\_In")){

input\_num=ADC\_In();

printf("\n\rSample from ADC: %d",input\_num);

}

else if(!strcmp(input\_str,"ADC\_Collect")){

printf("\n\rChannel to Open: ");

input\_num=UART\_InUDec();

printf("\n\rSampling Frequency: ");

freq=UART\_InUDec();

printf("\n\rNumber of Samples: ");

numSamples=UART\_InUDec();

ADC\_Collect(input\_num,freq,TestBuffer,numSamples);

}

else if(!strcmp(input\_str,"ADC\_Status")){

int i;

input\_num = ADC\_Status();

if(input\_num==0){

printf("\n\rStatus: Busy");

}else{

printf("\n\rStatus: Done");

printf("\n\rSamples Collected:");

for(i=0;i<numSamples;i++){

printf("\n\r%d",TestBuffer[i]);

}

}

}

else if(!strcmp(input\_str,"OS-RTP")){

printf("\n\rTimer to Read:");

input\_num = UART\_InUDec();

printf("\n\rCurrent Timer Period: ");

printf("%d",OS\_ReadTimerPeriod(input\_num));

}

else if(!strcmp(input\_str,"OS-RTV")){

printf("\n\rTimer to Read:");

input\_num = UART\_InUDec();

printf("\n\rCurrent Timer Value: ");

printf("%d",OS\_ReadTimerValue(input\_num));

}

else if(!strcmp(input\_str,"OS-CPT")){

printf("\n\rTimer to Clear:");

input\_num = UART\_InUDec();

OS\_ClearPeriodicTime(input\_num);

}

else if(!strcmp(input\_str,"OS-ST")){

printf("\n\rTimer to Stop:");

input\_num = UART\_InUDec();

OS\_StopThread(&PF1\_Toggle,input\_num);

}

else{

printf("\n\rInvalid Command. Try Again\n\r");

}

}

}

// initializes a new thread with given period and priority

int OS\_AddPeriodicThread(void(\*task)(void), int timer, unsigned long period, unsigned long priority)

{// period and priority are used when initializing the timer interrupts

int status;

int sr;

sr = StartCritical();

// initialize a timer specific to this thread

// each timer should be unique to a thread so that it can interrupt when

// it counts to 0 and sets the flag, this requires counting timers

status = 0;

status = OS\_TimerInit(task,timer, period, priority);

if(status == -1)

{

//printf("Error Initializing timer number(0-11): %d\n", timer);

}

EndCritical(sr);

return 0;

}

// Resets the 32-bit counter to zero

void OS\_ClearPeriodicTime(int timer)

{

switch(timer)

{

case 0: // TIMERA0

TIMER0\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA0

TIMER0\_TAV\_R = 0; // set Timer to 0

TIMER0\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA0

break;

case 1: // TIMERB0

TIMER0\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB0

TIMER0\_TBV\_R = 0; // set Timer to 0

TIMER0\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB0

break;

case 2: // TIMERA1

TIMER1\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA1

TIMER1\_TAV\_R = 0; // set Timer to 0

TIMER1\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA1

break;

case 3: // TIMERB1

TIMER1\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB1

TIMER1\_TBV\_R = 0; // set Timer to 0

TIMER1\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB1

break;

case 4: // TIMERA2

TIMER2\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA2

TIMER2\_TAV\_R = 0; // set Timer to 0

TIMER2\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA2

break;

case 5: // TIMERB2

TIMER2\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB2

TIMER2\_TBV\_R = 0; // set Timer to 0

TIMER2\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB2

break;

case 6: // TIMERA3

TIMER3\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA3

TIMER3\_TAV\_R = 0;// set Timer to 0

TIMER3\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA3

break;

case 7: // TIMERB3

TIMER3\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB3

TIMER3\_TBV\_R = 0; // set Timer to 0

TIMER3\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB3

break;

case 8: // TIMERA4

TIMER4\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA4

TIMER4\_TAV\_R = 0; // set Timer to 0

TIMER4\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA4

break;

case 9: // TIMERB4

TIMER4\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB4

TIMER4\_TBV\_R = 0; // set Timer to 0

TIMER4\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB4

break;

case 10: // TIMERA5

TIMER5\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA5

TIMER5\_TAV\_R = 0; // set Timer to 0

TIMER5\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA5

break;

case 11: // TIMERB5

TIMER5\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB5

TIMER5\_TBV\_R = 0; // set Timer to 0

TIMER5\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB5

break;

default:

break;

}

}

// Returns the number of bus cycles in a full period

unsigned long OS\_ReadTimerPeriod(int timer)

{

unsigned long busCyclePerPeriod = 0;

switch(timer)

{

case 0:

busCyclePerPeriod = TIMER0\_TAILR\_R;

break;

case 1:

busCyclePerPeriod = TIMER0\_TBILR\_R;

break;

case 2:

busCyclePerPeriod = TIMER1\_TAILR\_R;

break;

case 3:

busCyclePerPeriod = TIMER1\_TBILR\_R;

break;

case 4:

busCyclePerPeriod = TIMER2\_TAILR\_R;

break;

case 5:

busCyclePerPeriod = TIMER2\_TBILR\_R;

break;

case 6:

busCyclePerPeriod = TIMER3\_TAILR\_R;

break;

case 7:

busCyclePerPeriod = TIMER3\_TBILR\_R;

break;

case 8:

busCyclePerPeriod = TIMER4\_TAILR\_R;

break;

case 9:

busCyclePerPeriod = TIMER4\_TBILR\_R;

break;

case 10:

busCyclePerPeriod = TIMER5\_TAILR\_R;

break;

case 11:

busCyclePerPeriod = TIMER5\_TBILR\_R;

break;

default:

break;

}

return busCyclePerPeriod;

}

unsigned long OS\_ReadTimerValue(int timer)

{

unsigned long count = 0;

switch(timer)

{

case 0: // TimerA0

count = TIMER0\_TAR\_R; // Read value, x, stored in Timer0, 0 < x < TIMER0\_TAILR

break;

case 1: // TimerB0

count = TIMER0\_TBR\_R; // Read value, x, stored in Timer0, 0 < x < TIMER0\_TBILR

break;

case 2: // TimerA1

count = TIMER1\_TAR\_R; // Read value, x, stored in Timer1, 0 < x < TIMER1\_TAILR

break;

case 3: // TimerB1

count = TIMER1\_TBR\_R; // Read value, x, stored in Timer1, 0 < x < TIMER1\_TBILR

break;

case 4: // TimerA2

count = TIMER2\_TAR\_R; // Read value, x, stored in Timer2, 0 < x < TIMER2\_TAILR

break;

case 5: // TimerB2

count = TIMER2\_TBR\_R; // Read value, x, stored in Timer2, 0 < x < TIMER2\_TBILR

break;

case 6: // TimerA3

count = TIMER3\_TAR\_R; // Read value, x, stored in Timer3, 0 < x < TIMER3\_TAILR

break;

case 7: // TimerB3

count = TIMER3\_TBR\_R; // Read value, x, stored in Timer3, 0 < x < TIMER3\_TBILR

break;

case 8: // TimerA4

count = TIMER4\_TAR\_R; // Read value, x, stored in Timer4, 0 < x < TIMER4\_TAILR

break;

case 9: // TimerB4

count = TIMER4\_TBR\_R; // Read value, x, stored in Timer4, 0 < x < TIMER4\_TBILR

break;

case 10: // TimerA5

count = TIMER5\_TAR\_R; // Read value, x, stored in Timer5, 0 < x < TIMER5\_TAILR

break;

case 11: // timerB5

count = TIMER5\_TBR\_R; // Read value, x, stored in Timer5, 0 < x < TIMER5\_TBILR

break;

default:

break;

}

return count;

}

// launches all programs

//void OS\_LaunchAll(void(\*\*taskPtrPtr)(void))

//{

//// int i;

//// int timerN;

//// unsigned int TAEN\_TBEN;

//// for(i = 0; i <= timerCount; i++)

//// {

//// timerN = usedTimers[i]; // its the timerID 0 to 11

//// TAEN\_TBEN = ((timerN%2) ? TIMER\_CTL\_TBEN:TIMER\_CTL\_TAEN); // if timerN even => TAEN, timerN odd => TBEN

//// \*(timerCtrlBuf[timerN]) |= TAEN\_TBEN;

//// }

//

// // this assumes that the timers initialized in the same sequence as the tasks

// // i.e. usedTimer[i] was initialized for the functionPtr \*(taskPtrPtr[i])

// int i;

// for(i = 0; i <= timerCount; i++)

// {

// OS\_LaunchThread(taskPtrPtr[i],usedTimers[i]);

// }

//}

// enables interrupts in the NVIC vector table

void OS\_NVIC\_EnableTimerInt(int timer)

{

switch(timer)

{

case 0: // Timer0A

NVIC\_EN0\_R = NVIC\_EN0\_INT19;

break;

case 1: // Timer0B

NVIC\_EN0\_R = NVIC\_EN0\_INT20;

break;

case 2: // Timer1A

NVIC\_EN0\_R = NVIC\_EN0\_INT21;

break;

case 3: // Timer1B

NVIC\_EN0\_R = NVIC\_EN0\_INT22;

break;

case 4: // Timer2A

NVIC\_EN0\_R = NVIC\_EN0\_INT23;

break;

case 5: // Timer2B

NVIC\_EN0\_R = NVIC\_EN0\_INT24;

break;

case 6: // Timer3A

NVIC\_EN1\_R = NVIC\_EN1\_INT35;

break;

case 7: // Timer3B

NVIC\_EN1\_R = NVIC\_EN1\_INT36;

break;

case 8: // Timer4A

NVIC\_EN2\_R = NVIC\_EN2\_INT70;

break;

case 9: // Timer4B

NVIC\_EN2\_R = NVIC\_EN2\_INT71;

break;

case 10: // Timer5A

NVIC\_EN2\_R = NVIC\_EN2\_INT92;

break;

case 11: // Timer5B

NVIC\_EN2\_R = NVIC\_EN2\_INT93;

break;

}

}

void OS\_NVIC\_DisableTimerInt(int timer)

{

switch(timer)

{

case 0: // Timer0A

NVIC\_DIS0\_R = NVIC\_DIS0\_INT19;

break;

case 1: // Timer0B

NVIC\_DIS0\_R = NVIC\_DIS0\_INT20;

break;

case 2: // Timer1A

NVIC\_DIS0\_R = NVIC\_DIS0\_INT21;

break;

case 3: // Timer1B

NVIC\_DIS0\_R = NVIC\_DIS0\_INT22;

break;

case 4: // Timer2A

NVIC\_DIS0\_R = NVIC\_DIS0\_INT23;

break;

case 5: // Timer2B

NVIC\_DIS0\_R = NVIC\_DIS0\_INT24;

break;

case 6: // Timer3A

NVIC\_DIS1\_R = NVIC\_DIS1\_INT35;

break;

case 7: // Timer3B

NVIC\_DIS1\_R = NVIC\_DIS1\_INT36;

break;

case 8: // Timer4A

NVIC\_DIS2\_R = NVIC\_DIS2\_INT70;

break;

case 9: // Timer4B

NVIC\_DIS2\_R = NVIC\_DIS2\_INT71;

break;

case 10: // Timer5A

NVIC\_DIS2\_R = NVIC\_DIS2\_INT92;

break;

case 11: // Timer5B

NVIC\_DIS2\_R = NVIC\_DIS2\_INT93;

break;

}

}

void OS\_LaunchThread(void(\*taskPtr)(void), int timer)

{

int timerN;

unsigned int TAEN\_TBEN;

timerN = timer; // its the timerID (0 to 11)

OS\_NVIC\_EnableTimerInt(timer);

TAEN\_TBEN = ((timerN%2) ? TIMER\_CTL\_TBEN : TIMER\_CTL\_TAEN); // if timerN even => TAEN, timerN odd => TBEN

\*(timerCtrlBuf[timerN]) |= TAEN\_TBEN; // start timer for this thread, X in {0,1,2,3,4,5}, x in {A,B}

// ^^^ is \*(TIMERX\_CTRL\_PTR\_R) |= TIMER\_CTL\_TxEN and/or TIMERX\_CTL\_R |= TIMER\_CTL\_TxEN

//(\*taskPtr)(); // begin task for this thread Do this in the ISR

}

void OS\_StopThread(void(\*taskPtr)(void), int timer)

{

int timerN;

unsigned int TAEN\_TBEN;

timerN = timer; // its the timerID (0 to 11)

OS\_NVIC\_DisableTimerInt(timer);

TAEN\_TBEN = ((timerN%2) ? TIMER\_CTL\_TBEN : TIMER\_CTL\_TAEN); // if timerN even => TAEN, timerN odd => TBEN

\*(timerCtrlBuf[timerN]) &= ~TAEN\_TBEN; // start timer for this thread, X in {0,1,2,3,4,5}, x in {A,B}

// ^^^ is \*(TIMERX\_CTRL\_PTR\_R) |= TIMER\_CTL\_TxEN and/or TIMERX\_CTL\_R |= TIMER\_CTL\_TxEN

//(\*taskPtr)(); // begin task for this thread Do this in the ISR

}

// this configures the timers for 32-bit mode, periodic mode

//

int OS\_TimerInit(void(\*task)(void), int timer, unsigned long desiredFrequency, unsigned long priority)

{

int delay;

unsigned long cyclesPerPeriod;

// will fail if frequency is a decimal number close to 0 relative to the bus speed

cyclesPerPeriod = CLOCKSPEED\_80MHZ/desiredFrequency;

switch(timer)

{

case 0: //TimerA0

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R0; // activate timer0

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER0\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA0

TIMER0\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER0\_TAMR\_R = TIMER\_TAMR\_TAMR\_PERIOD;

TIMER0\_TAILR\_R = cyclesPerPeriod-1;

TIMER0\_TAPR\_R = 0; // set prescale = 0

TIMER0\_ICR\_R = TIMER\_ICR\_TATOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER0\_IMR\_R |= TIMER\_IMR\_TATOIM; // arm the timeout interrupt

NVIC\_PRI4\_R = (NVIC\_PRI4\_R & ~NVIC\_PRI4\_INT19\_M)|(priority << NVIC\_PRI4\_INT19\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER0\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA0, do this in OS\_Launch(.)

HandlerTaskArray[0] = task; // fill function pointer array w/address of task

break;

case 1: //TimerB0

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R0; // activate timer0

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER0\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB0

TIMER0\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER0\_TBMR\_R = TIMER\_TBMR\_TBMR\_PERIOD;

TIMER0\_TBILR\_R = cyclesPerPeriod-1;

TIMER0\_TBPR\_R = 0; // set prescale = 0

TIMER0\_ICR\_R = TIMER\_ICR\_TBTOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER0\_IMR\_R |= TIMER\_IMR\_TBTOIM; // arm the timeout interrupt

NVIC\_PRI5\_R = (NVIC\_PRI5\_R & ~NVIC\_PRI5\_INT20\_M)|(priority << NVIC\_PRI5\_INT20\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER0\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB0, do this in OS\_Launch(.)

HandlerTaskArray[1] = task; // fill function pointer array w/address of task

break;

case 2: //TimerA1

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R1; // activate timer1

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER1\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA1

TIMER1\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER1\_TAMR\_R = TIMER\_TAMR\_TAMR\_PERIOD;

TIMER1\_TAILR\_R = cyclesPerPeriod-1;

TIMER1\_TAPR\_R = 0; // set prescale = 0

TIMER1\_ICR\_R = TIMER\_ICR\_TATOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER1\_IMR\_R |= TIMER\_IMR\_TATOIM; // arm the timeout interrupt

NVIC\_PRI5\_R = (NVIC\_PRI5\_R & ~NVIC\_PRI5\_INT21\_M)|(priority << NVIC\_PRI5\_INT21\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER1\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA1, do this in OS\_Launch(.)

HandlerTaskArray[2] = task; // fill function pointer array w/address of task

break;

case 3: //TimerB1

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R1; // activate timer1

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER1\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB1

TIMER1\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER1\_TBMR\_R = TIMER\_TBMR\_TBMR\_PERIOD;

TIMER1\_TBILR\_R = cyclesPerPeriod-1;

TIMER1\_TBPR\_R = 0; // set prescale = 0

TIMER1\_ICR\_R = TIMER\_ICR\_TBTOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER1\_IMR\_R |= TIMER\_IMR\_TBTOIM; // arm the timeout interrupt

NVIC\_PRI5\_R = (NVIC\_PRI5\_R & ~NVIC\_PRI5\_INT22\_M)|(priority << NVIC\_PRI5\_INT22\_S);

//TIMER1\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB1, do this in OS\_Launch(.)

HandlerTaskArray[3] = task; // fill function pointer array w/address of tasks

break;

case 4: //TimerA2

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R2; // activate timer2

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER2\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA2

TIMER2\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER2\_TAMR\_R = TIMER\_TAMR\_TAMR\_PERIOD;

TIMER2\_TAILR\_R = cyclesPerPeriod-1;

TIMER2\_TAPR\_R = 0; // set prescale = 0

TIMER2\_ICR\_R = TIMER\_ICR\_TATOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER2\_IMR\_R |= TIMER\_IMR\_TATOIM; // arm the timeout interrupt

NVIC\_PRI5\_R = (NVIC\_PRI5\_R & ~NVIC\_PRI5\_INT23\_M)|(priority << NVIC\_PRI5\_INT23\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER2\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA2, do this in OS\_Launch(.)

HandlerTaskArray[4] = task; // fill function pointer array w/address of tasks

break;

case 5: //TimerB2

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R2; // activate timer2

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER2\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB2

TIMER2\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER2\_TBMR\_R = TIMER\_TBMR\_TBMR\_PERIOD;

TIMER2\_TBILR\_R = cyclesPerPeriod-1;

TIMER2\_TBPR\_R = 0; // set prescale = 0

TIMER2\_ICR\_R = TIMER\_ICR\_TBTOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER2\_IMR\_R |= TIMER\_IMR\_TBTOIM; // arm the timeout interrupt

NVIC\_PRI6\_R = (NVIC\_PRI6\_R & ~NVIC\_PRI6\_INT24\_M)|(priority << NVIC\_PRI6\_INT24\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER2\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB2, do this in OS\_Launch(.)

HandlerTaskArray[5] = task; // fill function pointer array w/address of tasks

break;

case 6: //TimerA3

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R3; // activate timer3

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER3\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA3

TIMER3\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER3\_TAMR\_R = TIMER\_TAMR\_TAMR\_PERIOD;

TIMER3\_TAILR\_R = cyclesPerPeriod-1;

TIMER3\_TAPR\_R = 0; // set prescale = 0

TIMER3\_ICR\_R = TIMER\_ICR\_TATOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER3\_IMR\_R |= TIMER\_IMR\_TATOIM; // arm the timeout interrupt

NVIC\_PRI8\_R = (NVIC\_PRI8\_R & ~NVIC\_PRI8\_INT35\_M)|(priority << NVIC\_PRI8\_INT35\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER3\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA3, do this in OS\_Launch(.)

HandlerTaskArray[6] = task; // fill function pointer array w/address of tasks

break;

case 7: //TimerB3

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R3; // activate timer3

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER3\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB3

TIMER3\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER3\_TBMR\_R = TIMER\_TBMR\_TBMR\_PERIOD;

TIMER3\_TBILR\_R = cyclesPerPeriod-1;

TIMER3\_TBPR\_R = 0; // set prescale = 0

TIMER3\_ICR\_R = TIMER\_ICR\_TBTOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER3\_IMR\_R |= TIMER\_IMR\_TBTOIM; // arm the timeout interrupt

NVIC\_PRI9\_R = (NVIC\_PRI9\_R & ~NVIC\_PRI9\_INT36\_M)|(priority << NVIC\_PRI9\_INT36\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER3\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB3, do this in OS\_Launch(.)

HandlerTaskArray[7] = task; // fill function pointer array w/address of tasks

break;

case 8: //TimerA4

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R4; // activate timer4

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER4\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA4

TIMER4\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER4\_TAMR\_R = TIMER\_TAMR\_TAMR\_PERIOD;

TIMER4\_TAILR\_R = cyclesPerPeriod-1;

TIMER4\_TAPR\_R = 0; // set prescale = 0

TIMER4\_ICR\_R = TIMER\_ICR\_TATOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER4\_IMR\_R |= TIMER\_IMR\_TATOIM; // arm the timeout interrupt

NVIC\_PRI17\_R = (NVIC\_PRI17\_R & ~NVIC\_PRI17\_INTC\_M)|(priority << NVIC\_PRI17\_INTC\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER4\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA4, do this in OS\_Launch(.)

HandlerTaskArray[8] = task; // fill function pointer array w/address of tasks

break;

case 9: //TimerB4

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R4; // activate timer4

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER4\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB4

TIMER4\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER4\_TBMR\_R = TIMER\_TBMR\_TBMR\_PERIOD;

TIMER4\_TBILR\_R = cyclesPerPeriod-1;

TIMER4\_TBPR\_R = 0; // set prescale = 0

TIMER4\_ICR\_R = TIMER\_ICR\_TBTOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER4\_IMR\_R |= TIMER\_IMR\_TBTOIM; // arm the timeout interrupt

NVIC\_PRI17\_R = (NVIC\_PRI17\_R & ~NVIC\_PRI17\_INTD\_M)|(priority << NVIC\_PRI17\_INTD\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER4\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB4, do this in OS\_Launch(.)

HandlerTaskArray[9] = task; // fill function pointer array w/address of tasks

break;

case 10: //TimerA5

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R5; // activate timer5

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER5\_CTL\_R &= ~TIMER\_CTL\_TAEN; // disable TimerA5

TIMER5\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER5\_TAMR\_R = TIMER\_TAMR\_TAMR\_PERIOD;

TIMER5\_TAILR\_R = cyclesPerPeriod-1;

TIMER5\_TAPR\_R = 0; // set prescale = 0

TIMER5\_ICR\_R = TIMER\_ICR\_TATOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER5\_IMR\_R |= TIMER\_IMR\_TATOIM; // arm the timeout interrupt

NVIC\_PRI23\_R = (NVIC\_PRI23\_R & ~NVIC\_PRI23\_INTA\_M)|(priority << NVIC\_PRI23\_INTA\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER5\_CTL\_R |= TIMER\_CTL\_TAEN; // enable TimerA5, do this in OS\_Launch(.)

HandlerTaskArray[10] = task; // fill function pointer array w/address of tasks

break;

case 11: //TimerB5

SYSCTL\_RCGCTIMER\_R |= SYSCTL\_RCGCTIMER\_R5; // activate timer5

delay = SYSCTL\_RCGCTIMER\_R; // allow time to finish activating

TIMER5\_CTL\_R &= ~TIMER\_CTL\_TBEN; // disable TimerB5

TIMER5\_CFG\_R = TIMER\_CFG\_32\_BIT\_TIMER; // configure for 32-bit mode

TIMER5\_TBMR\_R = TIMER\_TBMR\_TBMR\_PERIOD;

TIMER5\_TBILR\_R = cyclesPerPeriod-1;

TIMER5\_TBPR\_R = 0; // set prescale = 0

TIMER5\_ICR\_R = TIMER\_ICR\_TBTOCINT; // clear timeout flag, friendly since writing a 0 does nothing

TIMER5\_IMR\_R |= TIMER\_IMR\_TBTOIM; // arm the timeout interrupt

NVIC\_PRI23\_R = (NVIC\_PRI23\_R & ~NVIC\_PRI23\_INTB\_M)|(priority << NVIC\_PRI23\_INTB\_S); //clears PRI bits then shifts the mask into the appropriate place

//TIMER5\_CTL\_R |= TIMER\_CTL\_TBEN; // enable TimerB5, do this in OS\_Launch(.)

HandlerTaskArray[11] = task; // fill function pointer array w/address of tasks

break;

default:

return -1;

}

timerCount++; // used for launching the correct number of threads that were successfully initialized

usedTimers[timerCount] = timer; // store the timerID of which timer to launch

return 0;

}

void GPIO\_PortF\_Init(void)

{

unsigned long delay;

SYSCTL\_RCGCGPIO\_R |= SYSCTL\_RCGC2\_GPIOF;

delay = SYSCTL\_RCGCGPIO\_R;

GPIO\_PORTF\_DIR\_R |= 0x0F; // PF0-3 output

GPIO\_PORTF\_DEN\_R |= 0x0F; // enable Digital IO on PF0-3

GPIO\_PORTF\_AFSEL\_R &= ~0x0F; // PF0-3 alt funct disable

GPIO\_PORTF\_AMSEL\_R &= ~0x0F; // disable analog functionality on PF0-3

GPIO\_PORTF\_DATA\_R = 0x06;

}

void PF0\_Toggle(void)

{

GPIO\_PORTF\_DATA\_R ^= 0x01;

}

void PF1\_Toggle(void)

{

GPIO\_PORTF\_DATA\_R ^= 0x02;

}

void PF2\_Toggle(void)

{

GPIO\_PORTF\_DATA\_R ^= 0x04;

}

void PF3\_Toggle(void)

{

GPIO\_PORTF\_DATA\_R ^= 0x08;

}

#ifdef TESTING

int main(void)

{

int timer0;

int priority0;

int period0;

int timer1;

int priority1;

int period1;

int timer2;

int priority2;

int period2;

int timer3;

int priority3;

int period3;

timer0 = 0;

priority0 = 0;

period0 = 10;

timer1 = 2;

priority1 = 0;

period1 = 1000;

timer2 = 4;

priority2 = 0;

period2 = 1000;

timer3 = 6;

priority3 = 0;

period3 = 65;

GPIO\_PortF\_Init();

PLL\_Init();

//PF2\_Toggle();

OS\_AddPeriodicThread(&PF1\_Toggle, timer0, period0, priority0);

//PF2\_Toggle();

// OS\_AddPeriodicThread(&PF2\_Toggle, timer1, period1, priority1);

//OS\_AddPeriodicThread(&PF2\_Toggle, timer2, period2, priority2);

// OS\_AddPeriodicThread(&PF3\_Toggle, timer3, period3, priority3);

OS\_LaunchThread(&PF1\_Toggle, timer0);

// OS\_LaunchThread(&PF2\_Toggle, timer1);

// OS\_LaunchThread(&PF2\_Toggle, timer2);

// OS\_LaunchThread(&PF3\_Toggle, timer3);

while(1)

{;

}

return 0;

}

#endif

void Timer0A\_Handler(void)

{

TIMER0\_ICR\_R = TIMER\_ICR\_TATOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[0]))(); // start Timer0A task

}

void Timer0B\_Handler(void)

{

TIMER0\_ICR\_R = TIMER\_ICR\_TBTOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[1]))(); // start Timer0B task

}

void Timer1A\_Handler(void)

{

TIMER1\_ICR\_R = TIMER\_ICR\_TATOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[2]))(); // start Timer1A task

}

void Timer1B\_Handler(void)

{

TIMER1\_ICR\_R = TIMER\_ICR\_TBTOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[3]))(); // start Timer1B task

}

void Timer2A\_Handler(void)

{

TIMER2\_ICR\_R = TIMER\_ICR\_TATOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[4]))(); // start Timer2A task

}

void Timer2B\_Handler(void)

{

TIMER2\_ICR\_R = TIMER\_ICR\_TBTOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[5]))(); // start Timer2B task

}

void Timer3A\_Handler(void)

{

TIMER3\_ICR\_R = TIMER\_ICR\_TATOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[6]))(); // start Timer3A task

}

void Timer3B\_Handler(void)

{

TIMER3\_ICR\_R = TIMER\_ICR\_TBTOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[7]))(); // start Timer3B task

}

void Timer4A\_Handler(void)

{

TIMER4\_ICR\_R = TIMER\_ICR\_TATOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[8]))(); // start Timer4A task

}

void Timer4B\_Handler(void)

{

TIMER4\_ICR\_R = TIMER\_ICR\_TBTOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[9]))(); // start Timer4B task

}

void Timer5A\_Handler(void)

{

TIMER5\_ICR\_R = TIMER\_ICR\_TATOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[10]))(); // // start Timer5A task

}

void Timer5B\_Handler(void)

{

TIMER5\_ICR\_R = TIMER\_ICR\_TBTOCINT; // acknowledge interrupt flag

(\*(HandlerTaskArray[11]))(); // start Timer5B task

}

**Measurement Data:**

1. Estimated time to run periodic interrupt. The ISR was 6 assembly instructions and based on the assumption an instruction occurs every cycle it would be estimated at 6\*ClockPeriod, where Clock Period = 12.5ns. Estimated time is 75ns.

![](data:image/png;base64,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)

2) Measured time to run the ISR was 524 ns as can be seen by the Logic Analyzer snapshot. This deviation from the estimate can be attributed to the Reads and Writes in the ISR that access memory and take many more than a single cycle. Specifically, we do a write to acknowledge the interrupt. Also system operations occur even though the function is a ‘do-nothing’ task and it still requires pushing and popping the system state (Registers, LR, PC) before and after the function call. This measurement neglects the system overhead required to call the ISR since we profiled the ISR inside the Handler by setting and clearing a bit as the first and last instructions. This also introduces more time on the clearing of the bit since that requires an additional write to memory that is not part of the function of the ISR.

Time to execute the ISR

![C:\Users\Ken\Documents\EE445M\TEK00001.PNG](data:image/png;base64,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)

Time between ISRs

![C:\Users\Ken\Documents\EE445M\TEK00000.PNG](data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAeAAAADqCAIAAAD1WgEQAAARWElEQVR4nO2d4ZWjPA9Gp4QpISVMCVvKV8KWMKVsCVvKlEIHfD/Cm3WMbRwixIN87+HsSTAhQsCNRkvCxzRN0zR9fn4Up2lFYwgAAAxZPPsQbirfmoURNACAA7lne+S7uUxnlZ0ugPFNWKexPSfbTe9/9Lb//DprVY31b77jZr0CmydyZ+qy/Kf/vhpPcee+Go8CG4KuHbLFijtdIHtQ5KIpU6Yt3GxXZg9sTWQi6OOWL25sWy4cojV6/uzuXE9jZr/l26+91n5sCbrz8Xql6zN/87VHOGJAeiriTkFntcxLhXZnkd54i6l0DHSuqjhUXOE61MYCHJM1ap/6mzPbezyro3t0NJV26Powu9CufEvQte3cLM36F4CXyI7pqZTJfkG3F9sMY/dbPB43PmA2D6rGksVQ2+/IAdmgcdTVDsWePf6qPYorbMy8BNuCTqfi/PVKi4nOXpgOIWgrfATd3juNM+RVQa/X3CgOdnyibFYY6fvWVgL9gm6oYCqlfb2q9WtHF/T6NWlqahVH8WltSQRtRU8m3xT0tCWsgwS9qd1XBb1vE2DN+sBo7OjGTukX9OarGi+51n7sFXT7cWfKNvPFyfAmPoIurrY430HQmw/6V95ekmOyRr+g13N27JT2B3BkQX+WWhD9c9L1ZKvN3qI2Wlwb9NPQcXHOesf1nA+NvVM7PF56i0bkjTVn69lcebq2znXWtnpwak5sHzk9KuhfefbChpGutR+vFCsAwFAgaAAAUd4SdH6RBxMTExPTrukQQc/d3G4//Qsfh0gYcDTTn4/1dHZQAGUQ9IJIGA78PHN2OCeAneEqIOgFkTCcGVPQc+LoswMBaIGgF0TCcGZYQc/zjJ1BHwS9IBKGMwMKevr7ez2dHRRAGQS9IBKGM+MK+gGCBmEQ9IJIGM4gaAQNyiDoBZEwnBlX0HdH0+IAbRD0gkgYziBoBA3KIOgFkTCcGVDQc9rlwM6gDYJeEAnDmTEFPSd19NmBALRA0AsiYTgzrKDnecbOoA+CXhAJw5mRBQ2gD4JeEAnDGQQNoAyCXhAJwxkEDaAMgl4QCcMZBA2gDIJeEAnDGQQNoAyCXhAJwxkEDaAMgl4QCcMZBA2gDIJeEAnDGQQNoAyCXhAJwxkEDaAMgl4QCcMZBA2gDIJeEAnDGQQNoAyCXhAJwxkEDaAMgl4QCcMZBA2gDIJeEAnDGQQNoAyCXhAJwxkEDaAMgl4QCcMZBA2gjISgP75/nT6JCNp5q39+qsk/OxPz7J6NxnR2JhZOz4NUQk5PgkM2JAR93Ob1IxLG19eX59vVKmjnMGoQRoZIJISRcmgYRwn649d35yRixtvtpz/mQyfPra4J+vQknJKNGqcngYSMmw0q6DsiYThDDxpAGQS9IBKGMwgaQBkEvSAShjMIGkAZBL0gEoYzCBpAGQS9IBKGMwgaQBkEvSAShjMIGkAZBL0gEoYzCBpAGQS9IBKGAz/PnB0OAFRB0AsiYQAAPDjsm4RMTExMTG9Phwi68ZnwWR9lyHwo/ctAM8LLDdX+2NKJkKEYQ/MpLQ6RjR9kCEGbDyFohnyGZoUe9KG0N96NE8PoFLQnVw/D9r8rrp4NWwgjA0F7gKBTrh4Ggj4Owsg4StB3zt46lUQj6BSFMD7fOEQR9HEMG8b093c6zf8dosEraODyQXNIKZgz/f39T6D/OXoO3+KAtU2mXx+P6ZSQ9GmnCEGDOWtB3x2NoIOT2WT69bxncfSKJUV1RyNoMGeRcmLn+0QP2gOdHnTqnbMErbBTGj1oZ0ErZGMmjGdO60E/C5oedHwKFXRaRFNBr9j8I4MKGsxZC/o+H0EHp9ziOLWCFmczRQgazFlfxXEHQQeHHvSrUEGDDvSgPdDpQc8CV3Eo7JT2ddCeV3EoZGMmjGcUwuA66CGg3DNHLaVn/9oak+uEoEOhZpMASKX0ccbCINz3OIIOgpRNYqCTUuw8Jp+fH/SgLZHqQZ+OQhgxfosDQY+JgaANj2B4B51yLww6KbUV9O3283jweLx+2pgJ/byTPQQdBx2bhEEnpbmgs6+JvsJDuKmm10+zxzh6B/dUI2iYZyWbhEEnpU+CXl2+3X/Cpi7OxNEj6NQ4PdU3TKdX0Cmro0ridjKDDHHLq+LMxvG5uUKdW17lgn7+lZX+EzYT9KZkN6vsoushhQoa5lmp3AuDTkoLgk6flnj4t7On0VlB1+bj6BoIGuZZySZh0ElpocXxegX9kq+zx21BTzi6DoKGeVaySRh0UmrVg77TX1D3C3p6z0SBkehBn30AS1xyO3Md9DMKYcS8DvqNqzimvsvsisvQ4tgBFTTMs1K5FwadlPJFlTFB0HHQsUkYdFIaT9Drhnj/06usob3JPSDoOOjYJAw6KY0naOiBHrQx9KBTFMKI2YOGMaCCjoNOuRcGnZQi6DFB0HHQsUkYdFKKoMcEQcdBxyZh0Ekpgh4TetDG0INOUQiDHnRK48Ln9dPGTOjkzexRQcdBp9wLg05KTb6o0vPVwen5ixXFmdDJ+9lD0HHQsUkYdFJa/S2OvT8JPa2s0SPo9TW/6ZJIPANBwz90bBIGnZRaCTr73vamZDer7KLr4Y6EoOlBp9CDTlEII2APuu/HktLvrRW1u6PF0RA0jl4jIWjDIxjeQafcC4NOSk1+bvSBuaAnHF0CQcM/dGwSBp2Umvzc6Ev/Sdgp8aLN4Q6Chn/o2CQMOim1+rnR/svsiqUxLY5XOf8yu8Y93/xv3TbyUGcPeqih58Mzyj0JYRiooOOgU+6FQSel8QSd/QfmS0+vsob2JveAoOOgY5Mw6KQ0nqChBwQdBx2bhEEnpQh6TLgO2hiug05RCCPgddAwDFTQcdAp98Kgk1IEPSYIOg46NgmDTkoR9Jgg6Djo2CQMOim1FXTnpdDtmeAAPWhj6EGnKIRBDzolu8xr84uCtZngAxV0HHTKvTDopPSgCnqHoNeX/aZLInFbEHQcdGwSBp2U7hZ0rWVRfNAp6PZ8MARBx0HHJmHQSek+Qde6EwcJGkebQw/aGHrQKQphDNuDrkl5OkbQE44+ACroOOiUe2HQSek+QWdTOlR8UHzaL+hp9akAb4Kg46BjkzDopPSUy+yKpTEtDk8QdBx0bBIGnZTyRZUxoQdtDD3oFIUwhu1BQwCooOOgU+6FQSelCHpMEHQcdGwSBqmU4ujRuO9xBB0EKZvEQC2ln58fTONMi2PfFHTjnm+n35JuqKHOHvRQQ8+H54XvSchQyKH04Pz4/Cw79k1B16IBZ9TKvQCQUnADQQcHm5hDSsENBB0cbGIOKQU3jhI010GncB10ikIYMa6DNoQwUhTCoAc9BJR75pBScANBBwebmENKwQ0EHRxsYg4pBTfoQXtADzpFIQx60BmEkaIQBj3oIaDcM4eUghsIOjjYxBxSCm4g6OBgE3NIKbhBD9oDetApCmHQg84gjBSFMOhBDwHlnjmkFNxA0MHBJuaQUnADQQcHm5hDSsENetAe0INOUQiDHnQGYaQohEEPeggo98whpeBGcEErfBLOVNDPiISxGyro4yCMjBMELXunmZBD3PKqNtSgvUJuecWQz9B8dA+62OYT2fhBhhC0+RCCZujQoVSeVNDBhxB0bagBgmZIYWimB+3DiWHQgzaHHvRxEEZGcEEDlxyYQ0rBDa6D9oAKOkUhjM83DlEq6OMgjDSGc3rQ4AnlnjmkFNxA0MHBJuaQUnADQQcHm5hDSsENetAe0INOUQiDHnQGYaQohEEPeggo98whpeAGgg4ONjGHlIIbCDo42MQcUgpu0IP2gB50ikIY9KAzCCNFIQx60ENAuWcOKQU3EHRwsIk5pBTcQNDBwSbmkFJwgx60B/SgUxTCoAedQRgpCmHQgx4Cyj1zSCm4gaCDg03MIaXgBoIODjYxh5SCG9yTMPgQt7wqzmwcn5sr5JZXDB06lB6cVNDBodwzh5SCGwg6ONjEHFIKbiDo4GATc0gpuMF10B5wHXSKQhhcB51BGCkKYdCDHgLKPXNIKbiBoIODTcwhpeAGgg4ONjGHlIIb9KA9oAedohAGPegMwkhRCIMe9BBQ7plDSsENBB0cbGIOKQU3EHRwsIk5pBTcoAftAT3oFIUw6EFnEEaKQhj0oIeAcs8cUgpuIOjgYBNzSCm4gaCDg03MIaXgBj1oD+hBpyiEQQ86gzBSFMKgBz0ElHvmkFJwA0EHB5uYQ0rBDQQdHGxiDikFN7gnYfAh7klYnNk4PjdXyD0JGTp0KD04bQT9/b9f2fTxvXOqbQbs43b72b0vmIoTFbQtp+9Q6clQ0P9evLfF8fX1ZbvvAZvY8vX1RUpt4axvgKCDg01sQdDmcNY3qAp63bXYnFJB77vI1HxXKVzPOHMd9DMKYey+Dtpc0ArZmE8NIz3ryUYaQ6sH3bBwj6D3xcRnqTmUe7ZQQZvDWd/gBUFnDQ0EfQmwiS0I2hzO+gZU0MHBJrYgaHM46xscJWh60Cn0oFMUwqAHnUEPOkUhDHrQQ0C5ZwsVtDmc9Q0QdHCwiS0I2hzO+gYIOjjYxBYEbQ5nfQN60B7Qg05RCIMedAY96BSFMHb2oBsTFbQglHu2UEGbw1nfgK96Bweb2IKgzeGsb4Cgg4NNbEHQ5nDWNzhK0PSgU+hBpyiEQQ86gx50ikIYx/4e9L6YPn59M9lOlHu23FN6+m4NNp29V3XhllfBQdDmkFJwA0EHB5uYQ0rBDe5JGHyIexIWZzaOz80Vck9Chg4dSg9OKujgUO6ZQ0rBDQQdHGxiDikFNxB0cLCJOaQU3Di2B3321klczzhzHfQzCmHsvg56tha0QjZmwnhGIQx60ENAuWcOKQU3EHRwsIk5pBTcQNDBwSbmkFJwgx60B/SgUxTCoAedQRgpCmHQgx4Cyj1zSCm4gaCDg03MIaXgBoIODjYxh5SCG/SgPaAHnaIQBj3oDMJIUQiDHvQQUO6ZQ0rBDQQdHGxiDikFNxB0cLCJOaQU3KAH7QE96BSFMOhBZxBGikIYQ/SgFRI9I+hnrh4Ggj4OwshA0B4g6JSrh4Ggj4MwMk4QdGPjGTIf4pZX5kPc8oohn6EZQUcaut1++ie14DWHXkqpWvAMXX1oPk7QTExMTEzvT/aCBgCA40DQAACiIGjY4AYVSNQluOg+ugeMoKHFbaUheJAmh0TJ8tg119pH92gRNFS51gF9CvcUkShx0pr0QtxuNwQNVa54TDuDoC8BgoaA3I/p9Fqg9TKPmcUFaq86ij8f05/V22Uz33m6IhP09yv0bFAj8+sl05cU19O5huzxbmpHTjYnfbp+SXvhzuARNAQkO6Ybp337nHFy9MOkmWHTB+88LZEKumjebOarju5MncmOOELQjZnFgNcq71y4X9Dtz4Pau9cS8oik/6O0HwQNLVJBF+08Vc6c7HF2EE+rI97myHYTdFJW1wSdPX7MmZ4d3d6gfjtPFjviFEEXt2JzVetNXgefbuZD0O13z+Y0UlpbxhYEDS3MBd2e/y4+gn6W9UUFXZy/XuBNakVlbWb7JcWF1y9cf/ZPW4Ju56GnikfQcALpX4XZ0OY537NMsZrej5ugk6fHCbpWfmYaMtwR6wWsaNuw+I6N0XQNncJ9R9C1Nyp+PNQysA8EDS3agm5oInvc4wWDgztWi6O/fLbdEf1v3U+PDWsvqdl5MhK0SfFRDO99EDS0yI7pIj1/9216YfNduijq9aD/JDxY0DuyIbQjVqvqCabxklpsxdUW57wq6J7PvGLVjKDBj92X2fX8bTglJ5LZYZ32H9IHhpfZHdzi2P3H8vs7Yv34fXqCyTa5MVpUZ0PQjyU7Bd34ACi+UW0ZKxA0tLjipaPOHH2ZHZhQu8xu6hD0VPmYaSxjGDaChioIepOjv6gCJvBFFYjJFQ9rN/ixpEvAjyVBZMo/hgj83OhFuOg+ugf8fyUXMVUvjNl+AAAAAElFTkSuQmCC)

**Analysis and Discussion:**

1. What are the range, resolution, and precision of the ADC?

The range = a\*resolution\*precision + b, a,b are constants which scale the range from [-1, 1) to [-a+b, a+b). The range essentially maps from ADC values to the desired output value range (usually volts e.g. –Vdd to +Vdd). The range of our ADC is 0 to 4096 or 0 to 3.3 V

Precision is the total number of distinguishable values, precision = 2^n, where n = # of ADC bits used. The precision for the 12-bit TM4C123 ADC, the precision is 4096 alternatives.

Resolution is the smallest change that can be represented. For an n bit ADC that is 2^-n. The resolution of our ADC is 1/4096 or 0.8 mV.

1. List the ways the ADC conversion can be started. Explain why you chose the way you did.

The first approach to triggering an ADC sampling sequence is to use a periodic timer such as SysTick or one of the Timer modules to initiate an ADC conversion interrupt when the time rolls over. Another approach is to use software to trigger an ADC conversion by simply reading from the ADC FIFO built into hardware. This approach involves busy-wait synchronization as software triggers the conversion and waits for its completion. A third approach to triggering ADC conversion is edge-triggered interrupts from switches or external digital signals. This implementation is the same as for periodic timers. Our ADC modules implemented periodic timer ADC interrupts and software-triggered busy-wait synchronization. Periodic timer ADC interrupts should be used when an accurate sampling rate is necessary such as in digital acquisition systems. The software-triggered ADC is used for acquiring a single sample for debugging purposes.

1. The measured time to run the periodic interrupt can be measured directly by setting a bit high at the start of the ISR and clearing that bit at the end of the ISR. It could also be measured indirectly by measuring the time lost when running a simple main program that toggles an output pin. How did you measure it? Compare and contrast you method to these two.

We measured the time for the periodic interrupt by placing the debugging instruments in the ISR itself. This method is easier to implement and can be used when multiple interrupts are enabled. Toggling in the main program and measuring time lost includes the overhead of the time for a context switch but cannot be used if multiple interrupts are enabled.

1. Divide the time to execute one instance of the ISR by the total instructions in the ISR to get the average time to execute an instruction. Compare this to the 12.5 ns system clock period (80MHz).

524 ns/6 instructions = 87.33 ns/instruction

=> (87.33 ns/instruction)/ (12.5 ns/cycle) ~ 7 cycles/instruction

1. What are the range, resolution and precision of the SysTick Timer? i.e. answer the question relative to the NVIC\_ST\_CURRENT\_R register in the Cortex M4 core peripherals.

Precision is 2^24 distinguishable values. Resolution is 12.5 ns (1 clock cycle).

Range = (2^24)\*(12.5 ns) ~ 210 ms