# EE 371

# Lab 4: Building and Working with a Simple Microprocessor

Sean Leisle #1332740 Joe Jimenez #1242579 Keegan Griffee #1432621 The following report and its contents are solely our work.

| Reegan Diffee | 2/24/2017 |
|---------------|-----------|
|               | 2/24/2017 |
| Soe Dining    | 2/24/2017 |
| Sea My        |           |
|               | 2/24/2017 |

#### **Table of Contents**

- I. Abstract
- II. Introduction
- III. Process Used for Initializing the FPGA for Running C Applications
- IV. Scanning System with the Nios II Processor Integration Verilog Implementation
- V. Scanning System Control Application
- VI. The Nios II Processor Qsys Schematic
- VII. Scanning System with Processor Integration Signaltap Analysis
- VIII. C Applications Developed for the FPGA
- IX. Additional Subjects of Importance
- X. Anything the Lab Specification Didn't Think of
- XI. Discussion of Problems During Development
- XII. Summary and Conclusion

### I. Abstract

The goal of this lab was to integrate a processor with the onboard FPGA of the DE1-SoC and have them communicate using both verilog and C. We incorporated a microprocessor that ran compiled C code along with the gate arrays we programmed with verilog. After creating a few basic programs that utilized this interface of microprocessor and gate arrays, we then integrated our scanner subsystem from the previous lab to run commands from the processor. The tools used in this lab included iVerilog, Quartus II IDE, Nios II eclipse IDE, Qsys, the SignalTap II Logic Analyzer, PuTTY client for communication with our board, and the Linux operating system.

#### II. Introduction

The following lab report covers our implementation of several C programs that utilize the ARM Cortex-A9 processor that exists on the De1-SoC Cyclone V FPGA board used. In addition, we also used the Nios II soft processor that we designed in Qsys and programmed onto the DE1-SoC. The C programs we wrote exercised console input and output, controlling the DE1-SoC hardware from within programs, and interacting with gate arrays on the FPGA with the ARM and Nios II processors. It also covers the various paths we took in trying to accomplish the goal of getting the ARM microprocessor to communicate with the hardware as well as the Nios II processor. Finally, we discuss how we adjusted our previous implementation of the scanner subsystem from the previous lab to interact with the Nios II processor so that we could control the scanner subsystem from a program we wrote in C. We do not explicitly describe the implementation of this scanner subsystem in this lab report. Refer to the previous lab: Lab 3: Designing an Underwater Digital Scanning System for understanding this system. This report focuses on the process used for integrating the FPGA portion of the scanner system with the microprocessors along with the additional applications wrote in C to run on the DE1-SoC.

# III. Process Used for Initializing the FPGA for Running C Applications

#### Baremetal Approach (Failed):

When we first set out to accomplish integrating a processor to control the FPGA we attempted it with the ARM processor. We attempted to enable the ARM Cortex-A9 processor and program using the 'baremetal' approach. 'Baremetal' refers to programming a device without an operating system supporting it. The purpose of this approach was to use the Altera SoC EDS (Embedded Design Suite) software to target the Cyclone V FPGA to compile the C code we wrote and run it on the DE1-SoC. However, we attempted the tutorials for this process on three different computers and ran into the same roadblock each time when we tried to deploy the compiled code.

#### Linux Approach (Used for all non-scanner related C applications):

At this point we decided to switch to another promising approach of booting Linux on the FPGA and running C applications on Linux that would access the FPGA's hardware. In order to do this approach, we had to reconfigure the settings of the MSEL switches of the DE1-SoC board.



This configuration allows the ARM processor to program the DE1-SoC which is necessary for the Linux image to program the FPGA and enable access to the board's hardware on the DE1-SoC during bootup. We then flashed a compatible version of Linux onto a microSD card. We then inserted the microSD card into the DE1-SoC's microSD card reader so that when it booted it would boot from the microSD and run Linux. In order to access the Linux console on the DE1-SoC and be able to run our application we used a micro USB cable from the UART TO USB port on the DE1-SoC to our PC. This allowed us to run PuTTY on our PC to serial connect to the board and access the Linux command line. At this point we had to configure PuTTY to use the

COM serial line specified by the PC in the Device Manager tools. We then had to change the baud rate to 115200 bits per second, data bits set to 8 bits, stop bits set to 1 bit, and parity and flow control both set to none in device manager for the COM port the DE1-SoC was using our PC and set these same settings in PuTTY. Once these settings were configured we were able to run PuTTY and get the following window:

```
COM7 - PuTTY
EXT2-fs (mmcblk0p2): error: couldn't mount because of unsupported optional featu
res (244)
usb 1-1: new high-speed USB device number 2 using dwc2
usb 1-1: New USB device found, idVendor=0424, idProduct=2512
usb 1-1: New USB device strings: Mfr=0, Product=0, SerialNumber=0
hub 1-1:1.0: USB hub found
hub 1-1:1.0: 2 ports detected
usb 1-1.2: new high-speed USB device number 3 using dwc2
usb 1-1.2: New USB device found, idVendor=07d1, idProduct=3a09
usb 1-1.2: New USB device strings: Mfr=16, Product=32, SerialNumber=48
usb 1-1.2: Product: 11n adapter
usb 1-1.2: Manufacturer: ATHER
usb 1-1.2: SerialNumber: 12345
random: nonblocking pool is initialized
EXT4-fs (mmcblk0p2): 2 orphan inodes deleted
EXT4-fs (mmcblk0p2): recovery complete
EXT4-fs (mmcblk0p2): mounted filesystem with ordered data mode. Opts: (null)
VFS: Mounted root (ext4 filesystem) on device 179:2.
devtmpfs: mounted
Freeing unused kernel memory: 364K (806c7000 - 80722000)
init: ureadahead main process (52) terminated with status 5
Last login: Thu Jan 1 00:00:13 UTC 1970 on tty1
root@de1soclinux:~#
```

This verified we had successfully booted into Linux on the DE1-SoC and we could now issue commands through the terminal to run applications on our board using the ARM processor.

At this point we could now write C programs on our computers then cross compile them for ARM Linux (since compiling them on x86 machines results in unusable code on ARM processors) so that the resulting executable applications could be run on the DE1-SoC. We used the arm-linux-gnueabihf- toolchain provided when we downloaded the SoC EDS suite to compile our C programs and create executables to run on the DE1-SoC. In order to get our executable applications onto the DE1-SoC we just had to put them onto the microSD card which had a partition that was reachable by our Linux operating system even when we booted Linux on the DE1-SoC.

The difference when writing C programs for Linux vs. for 'baremetal' was that Linux uses virtual addresses, therefore we couldn't just deference a memory address to a peripheral like the LED's or Hex display from the DE1-SoC documentation. Instead, we had to call a function at the start of our C programs to get the offset caused by Linux and add the baremetal address of the peripheral to this offset, then we could properly

access the register that controlled the peripheral. This was our process for writing all of our C programs that controlled the DE1-SoC hardware.

However, we hit a final roadblock when we tried to program the FPGA with our scanning system and control it from a C program at the same time. We were able to program the FPGA with our control station during the bootup sequence of Linux using a raw binary file (.rbf) from our scanning control system top level design. Once we tried to run any C programs that utilized the hardware along with the FPGA it would throw a segmentation fault. We tried various ways to program the FPGA to get around this but came to the conclusion we needed to instead go back to the Nios II approach for this portion of the lab.

#### Nios II Approach (Used for scanner system processor integration):

After our many failed attempts at integrating the ARM processor with the scanning system, we quickly learned how to generate a soft processor (specifically the Nios II) from using the Quartus tool Qsys. We manually chose the input/output signals for our processor in order to control our scanner system. These signals included the ready to transfer signal for signalling the control station that the scanner was ready to transmit its data buffer, a start scanning signal to tell the scanner to start scanning, a transfer signal to actually initiate the transfer of the scanner's data buffer and finally a reset signal to reset the state of both the FPGA and the processor to return the system to a default starting state.

# IV. Scanning System with the Nios II Processor Integration Verilog Implementation

```
module lab4TopLevel (CLOCK_50, HEX0, HEX1, HEX2, HEX3, HEX4, HEX5, KEY, LEDR, SW, GPIO_0);
     input wire CLOCK_50; // 50MHz System Clock
input wire [9:0] SW; // Used for demo for flushing
     input wire [3:0] KEY; // Buttons
input wire [3:0] KEY; // Buttons
input wire [3:0] GPIO_0; // AC18 = dataOut, Y17 = clkOut, AD17 = readyToTransmitOut, AK16 = dataIn, AK18 = clkIn, AK19 = readyToTransmitIn
output wire [6:0] HEXO, HEX1, HEX2, HEX3, HEX4, HEX5; // Seven Segment Display
     output wire [9:0] LEDR; // LEDs
     wire [1:0] outFromTransfer, stationToScanner;
     wire [3:0] dataBufferScanner;
wire [7:0] dataBufferTransfer;
     wire [1:0] ps:
     wire [2:0] byteCounter;
     wire [7:0] commandBuffer, shiftReg;
     wire readyToTransfer;
wire startScanning;
     wire transfer:
     wire clkOut, rst;
wire [31:0] clkMain;
     parameter whichClock = 23;
assign rst = ~KEY[0];
     assign rst = kEY[0];
// Initialize clock divider
assign LEDR[9] = clkOut;
assign LEDR[9] = rst;
assign LEDR[1] = stationToScanner[1];
assign LEDR[6] = stationToScanner[0];
     assign clkOut = /*CLOCK_50; // CHANGE W
clock_divider cdiv (CLOCK_50, clkMain);
                                                         // DEBUG to see the signals actively working from the processor
     assign LEDR[3] = readyToTransfer;
assign LEDR[4] = startScanning;
     assign stationToScanner[1] = outFromTransfer[1] & SW[7]; // Demo for flushing the buffer
assign stationToScanner[0] = outFromTransfer[0] | startScanning;
     assign readyToTransfer = GPIO_0[0]; // Looks for the ready to transfer signal
// Initialize internals
seg7 h1 (dataBufferScanner, HEX1);
     seg7 h2 (dataBufferTransfer[3:0], HEX2);
seg7 h0 (4'b0, HEX0);
seg7 h3 (4'b0, HEX3);
       eq7 h4 (commandBuffer[3:0], HEX4);
     seg7 h5 (shiftReg[3:0], HEX5);
      // The Nios II processor with integration to receive the readyToTransfer signal // and transmit the startScanning signal, or start the transfer with the transfer signal
        // Also has a reset signal to reset the state of the processor
    nios_systemv3 u0 (
                                             (CLOCK 50).
           .clk clk
           .readytotransfer_export (readyToTransfer), // readytotransfer.export
           reset_reset_n (-rst), // reset_reset_
.startScanning_export (startScanning), // startScanning.export
.transfer_export (transfer) // transfer.export
           .reset reset n
                                                                                      reset.reset n
          .transfer export
                                                                                      transfer.export
     scanner localScanner (clkOut, rst, GPIO_0[6], stationToScanner, GPIO_0[1], GPIO_0[0], dataBufferScanner); // CHANGE 5 to 1 and 4 to 0 and 2 to 6
     transferCenterNew localTransfer (byteCounter, GPIO 0[5], rst, GPIO 0[4], transfer, GPIO 0[2], outFromTransfer, dataBufferTransfer, commandBuffer, shiftReg);
```

The top level implementation of the underwater digital scanning system. This handles assigning I/O for the FPGA and connecting our local scanner to the local control station. The outputs include a seven-segment display for the buffer level of the local scanner system on our board as well as GPIO pins assigned as outputs to send the necessary data and clock signals to a remote scanner from our control station. For inputs we used our newly integrated processor to control the system. The Nios II now can send the start scanning signal from the control station to a scanner as well as a transfer signal in order to transmit the data buffer from the scanner to the control station. It can also tell when the scanner is ready to transfer its data with the readyToTransfer signal or can receive a reset signal to set the system back to its starting state. We also have GPIO pins assigned as inputs to receive a clock and data from a remote scanner system not on our board. Finally we have the hex display as output for the various buffer levels in the system.

# V. Scanning System Control Application

```
Author: Keegan, Joe, Sean
         scannercontrol.c - Used to control the
        scanning system by command line utilizing
 #include "sys/alt_stdio.h"
#include "sys/unistd.h"
 #define readyToTransfer (volatile char *) 0x0009020
 #define transfer (volatile char *) 0x0009000
 #define startScanning (volatile char *) 0x0009010
⊟int main() {
   alt putstr ("Scanner Control Ready \n");
   char input = 'i'; // Idle/Don't print further output
   char nextState = 'i';
     // Always set scanning back to zero to prevent repeat scanning
     *startScanning = 0;
     // Always set transfer back to zero to prevent repeat transfering
     *transfer = 0;
     // Check if the scanner is ready to transfer it's data buffer
     if((*readyToTransfer &= 0x1) == 0x1) {
         alt_putstr("\n ready to transfer!\n");
     // Get the command from the user
     input = alt_getchar();
     // Use the command to determine the next state
     nextState = input;
     // Tell scanner to begin scanning
     if (nextState == 's') {
         // Prevent further scanning
         nextState = 'i';
         *startScanning = 1;
         // Reset the ready to transfer signal
         *readyToTransfer = 0;
         alt putstr("\n Start scanning...\n");
     // Tell the scanner to transfer it's data
     if (nextState == 't') {
         // Return to idle
         nextState = 'i';
         // Transfer the data
         *transfer = 1;
         alt putstr("\n Transfering data...\n");
   return 0;
```

C program used to control our scanner from console input. Can start the scanner, or tell the scanner to transmit it's data and informs us when the scanner is ready to transfer its data.

# VI. The Nios II Processor Qsys Schematic



#### **Processor Description:**

- -Operates on a 50 MHz clock
- -Has 16384 bytes of memory

#### I/O Signals Used By the Processor:

**startScanning** - Output signal that activates the scanner to begin actively scanning

**readyToTransfer** - Input signal that tells us that the scanner is ready to transfer data

**transfer** - Output signal that tells the scanner to begin transferring its data **reset** - Input signal that will reset the state of the processor

# VII. Scanning System with Processor Integration SignalTap Analysis



Above is a capture of the SignalTap Analysis of the working control system. The following are the controls and outputs for the system.

LED8 - Reset indicator

LED9 - Internal clock of scanner

Key 0 - Reset

AC18 - Data Out

Y17 - Clock Out

AD17 - Ready to transmit out

AK16 - Data In

AK18 - Clock In

AK19 - Ready to transmit in

Now Controlled by Processor

Activate scanner signal

Toggles the ability to flush the buffer (demo only)

Ready to Transfer Out

Begin transfer

# VIII. C Applications Developed for the FPGA A. Count Binary

Program that counts up by one repeatedly on both the FPGA's LEDs and on the 7-segment hex displays until it rolls over back to zero. The LED's represent the current count in binary and the hex displays the current count in decimal. Hex Drivers

```
#ifndef _HEX_HEADER_
   #define _HEX_HEADER_
4 #include <stdio.h>
 5 #include <stdlib.h>
 6 #include <unistd.h>
7 #include <fcntl.h>
8 #include <sys/mman.h>
9 #include <inttypes.h>
10
11 #define HW_REGS_BASE ( 0xff200000 )
12 #define HW_REGS_SPAN ( 0x00200000 )
13 #define HW_REGS_MASK ( HW_REGS_SPAN - 1 )
14 #define HEX_BASE 0x20
   #define HEX BASE TWO 0x30
16
   // Struct that represents one of the 7 segment hex displays
18 struct hex {
19
     uint32_t *address;
20
           uint8_t value;
           uint8_t index;
22
   };
23
24
   typedef struct hex *hex_t;
   // Create new hex with given index and virtualBase address
27
   hex_t allocateHex(uint8_t index, void *virtualBase);
28
29
   // Set value of the hex
   void setHexValue(hex_t hex, uint8_t value);
31
   // Return value of the hex
   uint8_t getHexValue(hex_t hex);
34
   #endif // _HEX_HEADER_
```

```
1 #include "hex.h"
  3 hex_t allocateHex(uint8_t index, void *virtualBase) {
              hex_t hex = (hex_t)malloc(sizeof(struct hex));
  4
  5
             if (hex == NULL) {
                     return NULL;
              }
  8
              hex->index = index;
              // Index < 4 has different register than index >= 4, gets the virtual address of register
 10
              if (index < 4) {
                      uint32_t *hexBase = (unsigned int *)(virtualBase + (( HEX_BASE ) & ( HW_REGS_MASK ) ));
                      hex->address = hexBase;
              } else {
 14
                      uint32_t *hexBase = (unsigned int *)(virtualBase + (( HEX_BASE_TWO ) & ( HW_REGS_MASK ) ));
                      hex->address = hexBase;
              }
 17
              setHexValue(hex, 0);
 18
              return hex;
 19 }
 20
 21 void setHexValue(hex_t hex, uint8_t value) {
              hex->value = value;
              uint32_t bits = 0;
 24
              // Choses correct bit pattern to make the number on the led display
              switch (value) {
                     case 0: bits = 0b00111111; break;
 27
                     case 1: bits = 0b00000110; break;
                      case 2: bits = 0b01011011; break;
                      case 3: bits = 0b01001111; break;
 29
                      case 4: bits = 0b01100110; break;
                      case 5: bits = 0b01101101; break;
                      case 6: bits = 0b01111101; break;
                      case 7: bits = 0b00000111; break;
                      case 8: bits = 0b01111111; break;
 34
                      case 9: bits = 0b01101111; break;
                      default: bits = 0b00111111; break;
            }
38
            // Calculate offset in the register
           int offset = (hex->index) % 4;
            // Shift bits to correct offset
41
           bits = bits << (offset * 8);
            // Clear the old value at that offset
42
            uint32_t clearer = 0x000000FF;
           clearer = clearer << (offset * 8);</pre>
            clearer = ~clearer;
46
           *(hex->address) = *(hex->address) & clearer;
47
            // Add in the bits of the value
            *(hex->address) = *(hex->address) | bits;
49 }
51 uint8_t getHexValue(hex_t hex) {
           return hex->value;
53 }
```

#### **Actual Counting program**

```
#include "hex.h"
 3 #define LED_BASE 0x0
 4 #define DELAY 10000000
 5 #define MAX_NUM 421
 6
 7 int main(int argc, char **argv) {
            void *virtual_base;
 9
            uint32_t *leds;
10
            int fd;
            // Open /dev/mem
            if( (fd = open( "/dev/mem", ( O_RDWR | O_SYNC ) ) ) == -1 ) {
14
                   printf( "ERROR: could not open \"/dev/mem\"...\n" );
                    return(1);
            }
17
            // Get virtual base address that maps to physical
            virtual_base = mmap( NULL, HW_REGS_SPAN, ( PROT_READ | PROT_WRITE ), MAP_SHARED, fd, HW_REGS_BASE );
            if( virtual_base == MAP_FAILED ) {
                    printf( "ERROR: mmap() failed...\n" );
                    close( fd );
23
                    return(1);
24
            }
25
            // Get address of the LED register
27
            leds = (unsigned int *)(virtual_base + (( LED_BASE ) & ( HW_REGS_MASK ) ));
28
            // Initialize hexes
30
            hex_t hexes[6];
            int i;
32
            for (i = 0; i < 6; i++) {
                    hexes[i] = allocateHex(i, virtual_base);
                    if (hexes[i] == NULL) {
                           return 1;
            }
```

```
int number = 0;
40
             int counter = 0;
41
             // Keeps counting in binary on the LEDs, and in decimal on the hex displays
42
43
             while (number < MAX_NUM) {
                     counter = (counter + 1) % DELAY;
                     if (counter == 0) {
45
                             number = (number + 1) % MAX_NUM;
46
                             setHexValue(hexes[0], number % 10);
47
                             setHexValue(hexes[1], (number / 10) % 10);
                             setHexValue(hexes[2], (number / 100) % 10);
49
                             *leds = number;
51
                     }
             }
54
             if( munmap( virtual_base, HW_REGS_SPAN ) != 0 ) {
                     printf( "ERROR: munmap() failed...\n" );
                     close( fd );
                     return(1);
             }
             // Free the hex structs
             for (i = 0; i < 6; i++) {
61
                     free(hexes[i]);
62
             }
             close( fd );
             return 0;
     }
```

# B. Lights and Switches

Program that controls the FPGA's LEDs based on the position of the switches on the FPGA. When a switch is flipped up then the LED corresponding to that switch becomes lit and when the switch is flipped down the LED corresponding to that switch turns off. However, if any of the FPGA's keys are pressed the system sees this as a reset and instead all LED's are turned off until the key is released.

```
1 #include <stdio.h>
   #include <unistd.h>
3 #include <fcntl.h>
4 #include <sys/mman.h>
6 #define HW_REGS_BASE ( 0xff200000 )
7 #define HW_REGS_SPAN ( 0x00200000 )
8 #define HW_REGS_MASK ( HW_REGS_SPAN - 1 )
9 #define LED_BASE 0x0
10 #define SWITCH_BASE 0x40
11 #define KEY_BASE 0x50
    int main(void) {
14
            /* Declare volatile pointers to I/O registers
            (volatile means that the locations will not be cached, * even in registers)
           volatile unsigned int * LED_ptr = NULL; // red LED address
18
            volatile unsigned int * SW_switch_ptr = NULL; // SW slider switch address // pushbutton KEY address
           volatile unsigned int * KEY_ptr = NULL; // Pushbutton Key Address
           void *virtual_base;
20
          int SW_value;
          int fd;
           volatile int delay_count;
24
            // Open /dev/mem
            if( (fd = open( "/dev/mem", ( O_RDWR | O_SYNC ) ) ) == -1 ) {
                    printf( "ERROR: could not open \"/dev/mem\"...\n" );
28
                   return( 1 );
30
       // get virtual addr that maps to physical
            virtual_base = mmap( NULL, HW_REGS_SPAN, ( PROT_READ | PROT_WRITE ), MAP_SHARED, fd, HW_REGS_BASE );
            if( virtual_base == MAP_FAILED ) {
                    printf( "ERROR: mmap() failed...\n" );
34
                    close( fd );
                   return(1);
            }
```

```
// Get the address that maps to the LEDs and other peripherals
            LED_ptr = (unsigned int *)(virtual_base + (( LED_BASE ) & ( HW_REGS_MASK ) ));
             SW_switch_ptr = (unsigned int *)(virtual_base + (( SWITCH_BASE ) & ( HW_REGS_MASK ) ));
41
42
             KEY_ptr = (unsigned int *)(virtual_base + (( KEY_BASE ) & ( HW_REGS_MASK ) ));
43
            while (1) {
                    // Get value of switches
45
                    SW_value = *(SW_switch_ptr);
47
                    // Reset on key press
                    if (*KEY_ptr != 0) {
48
49
                             *(LED_ptr) = 0;
                             while (*KEY_ptr != 0);
                    }
                    // Set LEDS to switch values
                    *(LED_ptr) = SW_value;
54
             }
             if( munmap( virtual_base, HW_REGS_SPAN ) != 0 ) {
                     printf( "ERROR: munmap() failed...\n" );
                    close( fd );
                    return(1);
61
             close( fd );
62
             return 0;
63 }
```

## C. Basic Hello World (Basic Console Output)

This program simply prints 'Hello World' to the console.

```
#include <stdio.h>
int main(int argc, char **argv) {
         printf("Hello, World!\n");
        return 0;
}
```

### D. Advanced Hello World (Console Input/Output)

This program first prints 'Hello World' to the console, then asks the user to input a 'G'. Once the user inputs a 'G' the FPGA hardware is activated so that when a switch is flipped up (from SW1 to SW9) the corresponding LED becomes lit up and doesn't light up if the switch is down. However, if SW0 is flipped up LED0 will

turn on and now switches that are in the downright position will cause their corresponding LED to light up and upright switches cause their LED to turn off.

```
1 #include <stdio.h>
 2 #include <unistd.h>
3 #include <fcntl.h>
4 #include <sys/mman.h>
6 #define HW_REGS_BASE ( 0xff200000 )
7 #define HW_REGS_SPAN ( 0x00200000 )
8 #define HW_REGS_MASK ( HW_REGS_SPAN - 1 )
9 #define LED_BASE 0x0
10 #define SWITCH_BASE 0x40
12 int main(void) {
          /* Declare volatile pointers to I/O registers
14
           (volatile means that the locations will not be cached, * even in registers)
           volatile unsigned int * LED_ptr = NULL; // red LED address
           volatile unsigned int * SW_switch_ptr = NULL; // SW slider switch address // pushbutton KEY address
           void *virtual_base;
           int SW_value;
          int fd;
           volatile int delay_count;
          // Open /dev/mem
24
          if( ( fd = open( "/dev/mem", ( O_RDWR | O_SYNC ) ) ) == -1 ) {
                   printf( "ERROR: could not open \"/dev/mem\"...\n" );
                   return(1);
          }
28
      // get virtual addr that maps to physical
           virtual_base = mmap( NULL, HW_REGS_SPAN, ( PROT_READ | PROT_WRITE ), MAP_SHARED, fd, HW_REGS_BASE );
           if( virtual base == MAP FAILED ) {
                   printf( "ERROR: mmap() failed...\n" );
                   close( fd );
34
                   return(1);
```

```
// Get the address that maps to the LEDs and other peripherals
             LED_ptr = (unsigned int *)(virtual_base + (( LED_BASE ) & ( HW_REGS_MASK ) ));
38
             SW_switch_ptr = (unsigned int *)(virtual_base + (( SWITCH_BASE ) & ( HW_REGS_MASK ) ));
41
             // Scan for input, G to continue
             char input;
42
43
             printf("hello world...\n");
             printf("To continue enter \'G\': ");
44
45
             scanf("%c", &input);
             getchar();
46
47
             while (input != 'G') {
                     printf("To continue enter \'G\': ");
                     scanf("%c", &input);
49
50
                     getchar();
             }
             while (1) {
                     // Get value of switches
54
                     SW_value = *(SW_switch_ptr);
                     // Invert if first switch is on
                     if (SW_value % 2 == 1) {
                             SW_value = ~SW_value + 1;
59
                    }
                    // Set LEDS to the value
                     *(LED_ptr) = SW_value;
             3
             if( munmap( virtual_base, HW_REGS_SPAN ) != 0 ) {
64
                     printf( "ERROR: munmap() failed...\n" );
                     close( fd );
                     return(1);
68
             close( fd );
70
             return 0;
71 }
```

### IX. Additional Subjects of Importance

None came to mind when writing and explaining the process we used in setting up the FPGA for the various programs and integration with the scanner system. One thing we unanimously agreed upon was a better set of tutorials need to be done in order to proceed with the ARM method of programming the FPGA. We almost got it to entirely work with ARM, however we ran into an eventual roadblock we couldn't overcome and made a last minute switch to using the Nios II cpu instead.

## X. Anything the Lab Specification Didn't Think of

We did not think of anything additional for this lab report that was not specified.

# XI. Discussion of Problems During Development

We ran into many problems during this lab involving the ARM processor approach. The first problem we ran into was when we attempted to program the DE1-SoC using the 'baremetal' approach using the SoC FPGA Embedded Design Suite (SoC EDS). Whenever we tried to deploy the code to our board and debug the application, the board was unrecognized by the development environment and we couldn't work around this problem.

We instead decided to utilize the other possible approach which was booting Linux and instead of writing baremetal C programs, we wrote programs that would go through the Linux operating system to get to the registers and peripherals we needed in order to access and control the DE1-SoC's hardware's behavior. This approach worked smoothly for our basic applications, however integrating our scanner control system onto the board with the processor interacting with it often halted our entire Linux operating system. After searching for another way to interface with the ARM processor, we found a tutorial on how to build a bridge between the ARM processor and the FPGA using Qsys. This approach also failed, however, so we decided to try a completely different approach.

We then decided to set up the Nios II soft processor on the FPGA and connect it to our scanner program. We managed to succeed with enabling the processor and sending control signals from a console to the processor which then controlled our scanner. However, we still found the processor was glitchy and resulted in some faulty behavior.

Due to time constraints and the amount of effort we put into trying to integrate the ARM processor we were unable to iron out all of the bugs the Nios II processor introduced which were present in our demo when certain sequences of commands to the console were issued. This also seemed to exacerbate the issues we had from our previous lab design and some of the failed state transitions.

In hindsight, we all agree that it would have been better to have just started with the Nios II approach and spent more time debugging our original scanner system when integrated with the processor. However, the insight we gained from learning new tools such as the SoC EDS, cross-compiling our C code for running on ARM, learning how to boot Linux onto the DE1-SoC while deciphering the DE1-SoC manual, using Qsys to design hardware bridges, and the process for writing Linux applications to control underlying hardware were invaluable and taught us more about general embedded system design than this lab originally entailed.

# XII. Summary and Conclusion

The product of this lab was a scanner control system that is now integrated with a Nios II soft microprocessor. We also developed various C programs that successfully ran on the DE1-SoC on the ARM processor and Linux operating system and ranged from printing simple console output, interacting with the user, determining the behavior of the hardware, and reacting to peripherals on the DE1-SoC. We also got to experience a variety of methods for programming and accessing the ARM processor on the DE1-SoC. This even involved learning how to boot and run Linux on the DE1-SoC. We learned both the 'baremetal' approach using no operating system and the Linux operating system approach. This taught us how to program hardware even with the abstraction of an operating system. This taught us valuable lessons in embedded design and perseverance when one route doesn't work. Overall, we understand even

more about the hardware on the FPGA and the process behind how software runs on it and affects its behavior.