

### Universidade Estadual de Campinas

### FACULDADE DE ENGENHARIA MECÂNICA

# ES670 - Projeto de Sistemas Embarcados

## Relatório - Projeto Prático Parte 1 Requisitos de Teclado, LEDs e Display de Sete Segmentos

RA

Nome: Daniel Dello Russo Oliveira 101918 Davi Rodrigues 116581

### 1 Objetivo

O objetivo do projeto é, de maneira incremental, implementar no target os requisitos apresentados no roteiro[1] inicialmente desenvolvendo o modelo e depois implementando cada requisito. Estes requisitos são referentes à configuração e implementação de entradas de teclado, acionamento de LEDs, display de sete segmentos, protocolo de comunicação, display LCD, medição de velocidade de rotação, PWM, ADC e Controlador.

### 2 Modelagem

Utilizando o Rational Rhapsody Modeler e tomando como base os requisitos propostos mostrados na figura 1, complementamos o modelo inicial[2] (requisitos de teclado e LEDs) adicionando um bloco ao modelo referente aos displays de sete segmentos (REQ1C), conforme mostrado na figura 3. Adicionamos também alguns blocos auxiliares relacionados ao gerenciamento de pinos GPIO e a interrupções periódicas, que foram utilizados para nossa implementação do display de sete segmentos e do buzzer. Ao tratar o gerenciamento do display e do buzzer através de interrupções, livramos a thread principal para que essa lide com outros problemas sem precisar se preocupar com a atualização periódica dos displays.



Figura 1: Diagrama de requisitos



Figura 2: Diagrama de pacotes



Figura 3: Diagrama de definição de blocos

O bloco de  $GPIO\_hal$  tem operações para desbloquear o clock para uma porta, inicializar um pino em uma dada direção, escrever em um pino, escrever em um conjunto de pinos da mesma porta e ler a entrada em um pino.

O bloco  $pit\_hal$  tem tem operações para inicializar o PIT, criar interrupções periódicas em um dos dois timers disponíveis, desativar as interrupções em um timer e marcar uma interrupção como tratada (deve ser feito pelos tratamentos de interrupção). Além disso esse bloco tem mais uma operação chamada  $PIT\_IRQHandler$  que trata as interrupções do PIT, essa operação precisa ser visível para o linker, mas não deve ser chamada pelo usuário. É importante ressaltar que o clock do PIT é o  $bus\_clock$  que no nosso caso é de 20MHz

As operações do bloco sevenseg\_hal cobrem a inicialização dos displays, seleção manual de quais segmentos estarão ativos (feita através da passagem de um vetor com os segmentos desejados), ativação manual de um dos displays (desativando todos os outros), conversão de dígito hexadecimal ou decimal em vetor de segmentos (para ser passado para a função de seleção de segmentos) e impressão automática de um valor hexadecimal ou decimal através das interrupções de timer.

O bloco do  $buzzer\_hal$  também ganhou duas novas operações, uma para criar uma onda quadrada no buzzer com um certo período (e duty cycle de 50%), através de interrupções de timer, e outra para remover essa onda.

### 3 Diagramas Esquemáticos



Figura 4: Esquema teclado e LEDs



Figura 5: Esquema sete segmentos

Os pinos apresentados acima, devem ser mapeados para os da placa FRDM-KL25Z através do mapeamento apresentado na figura 6



Figura 6: Mapeamento entre pinos dos esquemáticos e da placa FRDM-KL25Z

Como pode ser visto na figura 5, é necessário fazer um gerenciamento dos pinos PTC0 a PTC7 para selecionar os segmentos que serão ativados e PTC10 a PTC13 para selecionar quais displays estarão ativos. Para isso, é preciso alternar qual display está ativo e fazer a mudança nos segmentos para que cada display esteja mostrando um valor diferente. É importante lembrar que a frequência dessa alternância seja escolhida de modo que o olho humano não perceba que os displays estão ligando e desligando. Afim de garantir que essa alternância funcionará apropriadamente nós utilizamos o módulo PIT para gerar uma interrupção a cada 3.125ms, conforme sugerido na aula 6 [4].

#### 4 Matriz de Rastreabilidade

A matriz de rastreabilidade apresentada na tabela 1 relaciona cada um dos requisitos com a sua implementação.

Tabela 1: Matriz de Rastreabilidade

| ID do Requisito | Implementação                                                     |
|-----------------|-------------------------------------------------------------------|
| REQ1A           | ledswi_hal.c                                                      |
|                 | - void ledswi_initLedSwitch(char cLedNum, char cSwitchNum)        |
|                 | - switch_status_type_e ledswi_getSwitchStatus(char cSwitchNum)    |
| REQ1B           | ledswi.c                                                          |
|                 | - void ledswi_initLedSwitch(char cLedNum, char cSwitchNum)        |
|                 | - void ledswi_setLed(char cLedNum)                                |
|                 | - void ledswi_clearLed(char cLedNum)                              |
| REQ1C           | sevenSeg.c                                                        |
|                 | - void sevenseg_init(void)                                        |
|                 | - void sevenseg_setSegs(seven_segment_seg_type_e* epSet_segments) |
|                 | - void sevenseg_setDisp(seven_segment_disp_type_e eDisplay)       |
|                 | - void sevenseg_printDec(unsigned int uiDec)                      |
|                 | - void sevenseg_printHex(unsigned int uiHex)                      |

#### 5 Notas

#### 5.1 Gerenciamento de GPIO e macros

Detectamos logo no início do projeto um defeito estrutural no código fornecido quando lidando com GPIO: o identificador da porta e o número do pino utilizado eram referenciados em diversos locais diferentes do código dificultando de maneira agravante mudanças na configuração de hardware. Para resolver isso inicialmente pensamos em utilizar o arquivo  $fsl\_gpio\_hal.h$  da biblioteca da FRDM-Kl25Z, mas isso não nos foi permitido. Como calcular as posições na memória de cada registrador seria reimplementar a biblioteca, escolhemos por criar macros que geram o mesmo estilo de código utilizado no exemplo fornecido através do operador de concatenação (##) do pré processador. Esse operador apresenta algumas particularidades, a principal sendo que macros que o utilizam em seu corpo não tem seus argumentos expandidos [5]. Para circular essa dificuldade criamos uma outras macros que funcionam como uma wrappers para essas macros, fazendo assim que seus argumentos sejam expandidos antes da chamada da concatenação.

As macros que fazem a concatenação propriamente ditas não devem ser chamadas pelo usuário (sendo identificadas por um \_ no início de seus nomes).

Outra dificuldade relacionada a esse módulo é que a expansão dos argumentos das macros não para quando chega em alguma *token* não definida, no caso o identificador das portas (A,B,C,D,E). Para contornar esse problema utilizamos *typedefs* para definir esses identificadores como *tokens* válidas.

#### 5.2 Interrupções por timer

Outro problema que enfrentamos foi com a implementação das interrupções por timer através do PIT, notavelmente pela dificuldade de encontrar uma documentação clara sobre o NVIC e pois a documentação fornecida para o PIT inverte a endianness dos registradores em relação aos registradores de GPIO.

#### 5.3 Leitura de entradas digitais

Nesse laboratório tivemos dificuldades relacionadas à leitura do estado dos botões. O problema encontrado ocorreu pois quando o desbloqueamos o *clock* para uma porta do controlador no código fornecido estávamos desabilitando o *clock* para todas as outras portas. Como o *clock* só afeta significativamente os pinos configurados como *input*, se o módulo *ledswi\_hal* fosse inicializado por último (como é feito no código do professor) o problema não seria notado.

#### 5.4 Estilo de Documentação

Não conseguimos nos adaptar ao estilo de comentários sugerido pelo professor, que nos parece introduzir uma quantidade desnecessária de burocracia. Formatar todos os comentários para caber dentro daquele quadrado tomava mais tempo que o planejamento e a implementação do código e muita informação redundante estava sendo inserida (bastava ler a interface da função para saber seu nome e o tipo de seus argumentos). Escolhemos substituir todos os comentários do código pelo padrão javadoc, com o qual estamos mais familiarizados, que é capaz de documentar de maneira eficiente o código.

#### 5.5 Outros

Também tivemos dificuldades com a instalação do Rhapsody Rational Modeler no Linux (através do Wine), pois ele necessita da instalação das seguintes dlls nativas do Windows: comctl32, msvcirt e riched20.

É relevante lembrar de utilizar o modificador *volatile* para variáveis que serão modificadas durante o tratamento de interrupções, dessa maneira o otimizador sabe que não deve alterar comandos que envolvem essa variável.

### 6 Referências

- [1] Roteiro de Laboratório Semanas 04 e 05 (disponibilizado para os alunos)
- [2] Projeto do Modelo Inicial do Sistema (disponibilizado para os alunos)
- [3] Código Fonte Inicial em Linguagem C (disponibilizado para os alunos)
- [4] Notas de Aula Semanas 06 (disponibilizado para os alunos)
- [5] The C Preprocessor (Concatenation) https://gcc.gnu.org/onlinedocs/cpp/Concatenation.html#Concatenation

### 7 Apêndice

Listagem dos códigos fonte:

### 7.1 ../Sources/LedSwi/ledswi hal.h

```
2 /* File name:
                    ledswi_hal.h
                                                                 */
3 /* File description: Header file containing the function/methods
                                                                 */
                   prototypes of ledswi.c
                                                                 */
                     dloubach
5 /* Author name:
                                                                 */
6 /* Creation date: 09jan2015
                                                                 */
7 /* Revision date: 13abr2016
                                                                 */
10 #ifndef SOURCES_LEDSWI_LEDSWI_HAL_H_
11 #define SOURCES_LEDSWI_LEDSWI_HAL_H_
13 #define MAX_LED_SWI
                        04
15 typedef enum
     SWITCH_OFF,
    SWITCH_ON
19 } switch_status_type_e;
22 * As the hardware board was designed with LEDs/Switches sharing
^{23} * the same pins, this method configures how many LEDS and switches
24 * will be available for the application
* Cparam cLedNum num of LEDs
* Oparam cSwitchNum num of Switches (cLedNum + cSwitchNum <= MAX_LED_SWI)
28 void ledswi_initLedSwitch(char cLedNum, char cSwitchNum);
32 * set the led ON
33 * Oparam cLedNum which LED {1..4}
35 void ledswi_setLed(char cLedNum);
```

```
38 /**
39 * set the led OFF
40 * Oparam cLedNum which LED {1..4}
41 */
42 void ledswi_clearLed(char cLedNum);
43
44
45 /**
46 * return the switch status
47 * Oparam cSwitchNum which switch
48 * Oreturn If the switch is ON or OFF
49 */
50 switch_status_type_e ledswi_getSwitchStatus(char cSwitchNum);
51
52
53 #endif /* SOURCES_LEDSWI_LEDSWI_HAL_H_ */
```

#### 7.2 ../Sources/LedSwi/ledswi hal.c

```
2 /* File name:
                   ledswi_hal.c
_3 /* File description: This file has a couple of useful functions to _*/
                    control LEDs and Switches from peripheral board */
5 /* Author name:
                   dloubach
6 /* Creation date:
                                                                */
                   20 jan2015
7 /* Revision date:
                    13abr2016
                                                                */
10 #include "ledswi_hal.h"
#include "KL25Z/es670_peripheral_board.h"
13 #undef USING_OPENSDA_DEBUG
14
16 * As the hardware board was designed with LEDs/Switches sharing
17 * the same pins, this method configures how many LEDS and switches
18 * will be available for the application
19 * @param cLedNum num of LEDs
20 * Oparam cSwitchNum num of Switches (cLedNum + cSwitchNum <= MAX_LED_SWI)
22 void ledswi_initLedSwitch(char cLedNum, char cSwitchNum)
     /* un-gate port clock*/
     SIM_SCGC5 |= SIM_SCGC5_PORTA(CGC_CLOCK_ENABLED);
     /* set pin as gpio */
28 #ifndef USING_OPENSDA_DEBUG
     PORTA_PCR1 = PORT_PCR_MUX(LS1_ALT);
     PORTA_PCR2 = PORT_PCR_MUX(LS2_ALT);
31 #endif
     PORTA_PCR4 = PORT_PCR_MUX(LS3_ALT);
32
     PORTA_PCR5 = PORT_PCR_MUX(LS4_ALT);
34
35
     /* check if the number to configured is according to
     hardware dev kit */
37
     if((cLedNum + cSwitchNum) <= MAX_LED_SWI)</pre>
         /* max number of peripherals to configure is ok, carry on */
         switch(cSwitchNum)
41
         {
42
```

```
case 0:
43
44
                   /* no switches in system configuration */
                   /* all leds */
45
                   GPIOA_PDDR |= GPIO_PDDR_PDD(LS1_DIR_OUTPUT | LS2_DIR_OUTPUT |
46
      LS3_DIR_OUTPUT | LS4_DIR_OUTPUT);
                   break;
48
               case 1:
49
                   /* just 1 switch */
50
                   GPIOA_PDDR |= GPIO_PDDR_PDD(LS2_DIR_OUTPUT | LS3_DIR_OUTPUT |
51
      LS4_DIR_OUTPUT);
                   GPIOA_PDDR &= ~GPIO_PDDR_PDD (LS1_DIR_INPUT);
52
53
                   break;
54
55
              case 2:
                   /* just 2 switches */
                   GPIOA_PDDR |= GPIO_PDDR_PDD(LS3_DIR_OUTPUT | LS4_DIR_OUTPUT);
                   GPIOA_PDDR &= ~GPIO_PDDR_PDD(LS1_DIR_INPUT | LS2_DIR_INPUT);
58
                   break;
59
              case 3:
61
                   /* 3 switches */
62
                   GPIOA_PDDR |= GPIO_PDDR_PDD(LS4_DIR_OUTPUT);
                   GPIOA_PDDR &= ~GPIO_PDDR_PDD(LS1_DIR_INPUT | LS2_DIR_INPUT |
64
      LS3_DIR_INPUT);
                   break;
65
              case 4:
67
                   /* 4 switches */
68
                   GPIOA_PDDR &= ~GPIO_PDDR_PDD(LS1_DIR_INPUT | LS2_DIR_INPUT |
      LS3_DIR_INPUT | LS4_DIR_INPUT);
                   break;
70
          } /* switch(cSwitchNum) */
71
      } /* if((cLedNum + cSwitchNum) <= MAX_LED_SWI) */</pre>
73
74
75 }
76
77
78
80 * set the led ON
81 * Operam cLedNum which LED {1..4}
82 */
```

```
83 void ledswi_setLed(char cLedNum)
       /* sanity check */
       if(cLedNum <= MAX_LED_SWI)</pre>
           switch(cLedNum)
           {
89
                case 1:
90
                    GPIOA_PSOR = GPIO_PSOR_PTSO( (0x01U << LS1_PIN) );</pre>
91
                    break;
92
                case 2:
93
                    GPIOA_PSOR = GPIO_PSOR_PTSO( (0x01U << LS2_PIN) );</pre>
94
                    break;
                case 3:
96
97
                    GPIOA_PSOR = GPIO_PSOR_PTSO( (0x01U << LS3_PIN) );
                    break;
                case 4:
                    GPIOA_PSOR = GPIO_PSOR_PTSO( (0x01U << LS4_PIN) );</pre>
100
                    break;
101
           } /* switch(cLedNum) */
102
103
       } /* if(cLedNum <= MAX_LED_SWI) */</pre>
104
105 }
106
107
108
110 * set the led OFF
113 void ledswi_clearLed(char cLedNum)
114 {
115
       /* sanity check */
       if(cLedNum <= MAX_LED_SWI)</pre>
       {
117
           switch(cLedNum)
118
119
                case 1:
120
                    GPIOA_PCOR = GPIO_PCOR_PTCO( (0x01U << LS1_PIN) );</pre>
121
                    break:
122
                case 2:
                    GPIOA_PCOR = GPIO_PCOR_PTCO( (0x01U << LS2_PIN) );</pre>
124
                    break;
125
                case 3:
126
```

```
GPIOA_PCOR = GPIO_PCOR_PTCO( (0x01U << LS3_PIN) );</pre>
127
128
                case 4:
129
                     GPIOA_PCOR = GPIO_PCOR_PTCO( (0x01U << LS4_PIN) );</pre>
130
131
            } /* switch(cLedNum) */
132
133
       } /* if(cLedNum <= MAX_LED_SWI) */</pre>
134
135 }
136
137
138
139 /**
140 * return the switch status
141 * Oparam cSwitchNum which switch
_{142} * Oreturn If the switch is ON or OFF
143 */
144 switch_status_type_e ledswi_getSwitchStatus(char cSwitchNum)
145 {
        switch_status_type_e sstReturn = SWITCH_OFF;
146
147
       /* sanity check */
148
       if(cSwitchNum <= MAX_LED_SWI)</pre>
150
            switch(cSwitchNum)
151
            {
152
153
                case 1:
                     if(SWITCH_ON == ((GPIOA_PDIR & LS1_DIR_INPUT) >> LS1_PIN) )
154
                         sstReturn = SWITCH_ON;
155
                    break;
157
                case 2:
158
                     if (SWITCH_ON == ((GPIOA_PDIR & LS2_DIR_INPUT) >> LS2_PIN) )
159
                          sstReturn = SWITCH_ON;
                    break;
161
162
                case 3:
163
                     if(SWITCH_ON == ((GPIOA_PDIR & LS3_DIR_INPUT) >> LS3_PIN) )
164
                         sstReturn = SWITCH_ON;
165
                    break:
166
                case 4:
168
                     if(SWITCH_ON == ((GPIOA_PDIR & LS4_DIR_INPUT) >> LS4_PIN) )
169
                         sstReturn = SWITCH_ON;
170
```

### 7.3 ../Sources/Mcg/mcg hal.h

```
2 /* File name: mcg_hal.h
3 /* File description: Header file containing the functions/methods */
            interfaces for handling the Multipurpose clock st/
5 /*
              generator module
                                             */
                                             */
6 /* Author name:
             dloubach
7 /* Creation date: 21out2015
                                             */
8 /* Revision date: 25fev2016
                                             */
11 #ifndef SOURCES_MCG_HAL_H_
12 #define SOURCES_MCG_HAL_H_
15 /* Method name: mcg_clockInit */
16 /* Method description: main board clk configuration */
17 /* Input params: n/a
18 /* Output params: n/a
20 void mcg_clockInit(void);
21
24 #endif /* SOURCES_MCG_HAL_H_ */
```

## $7.4 \quad ../Sources/Mcg/mcg\_hal.c$

```
2 /* File name: mgc_hal.c
3 /* File description: Multipurpose clk generator hardware abstraction */
                    layer. Enables the clock configuration
5 /*
6 /*
                   Modes of Operation
                                                              */
                    FLL Engaged Internal (FEI) = DEFAULT
7 /*
                                                              */
                   FLL Engaged External (FEE)
8 /*
9 /*
                   FLL Bypassed Internal (FBI)
                                                              */
10 /*
                   FLL Bypassed External (FBE)
                   PLL Engaged External (PEE)
11 /*
                   PLL Bypassed External (PBE)
12 /*
                                                              */
13 /*
                   Bypassed Low Power Internal (BLPI)
                                                              */
14 /*
                   Bypassed Low Power External (BLPE)
15 /*
                    Stop
16 /*
17 /*
                   For clock definitions, check the chapter
                                                             */
                    5.4 Clock definitions from
19 /*
                   KL25 Sub-Family Reference Manual
                                                              */
20 /*
                                                              */
21 /* Author name: dloubach
                                                              */
22 /* Creation date: 21out2015
                                                              */
23 /* Revision date: 13abr2016
                                                              */
26 #include "mcg_hal.h"
28 /* systems include */
29 #include "fsl_smc_hal.h"
30 #include "fsl_port_hal.h"
31 #include "fsl_clock_manager.h"
33 /* EXTALO PTA18 */
34 #define EXTALO_PORT
                                 PORTA
35 #define EXTALO_PIN
                                  18 U
36 #define EXTALO_PINMUX
                                 kPortPinDisabled
38 /* XTALO PTA19 */
39 #define XTALO_PORT
                                 PORTA
40 #define XTALO_PIN
                                 19 U
41 #define XTALO_PINMUX
                                  kPortPinDisabled
```

```
43 /* OSCO configuration */
44 #define OSCO_INSTANCE
                                        0 U
45 #define OSCO_XTAL_FREQ
                                        8000000U /* 08 MHz*/
46 #define OSCO_SC2P_ENABLE_CONFIG
                                       false
47 #define OSCO_SC4P_ENABLE_CONFIG
                                       false
48 #define OSCO_SC8P_ENABLE_CONFIG
                                        false
49 #define OSCO_SC16P_ENABLE_CONFIG
                                        false
50 #define MCG_HGOO
                                        kOscGainLow
51 #define MCG_RANGEO
                                        kOscRangeVeryHigh
52 #define MCG_EREFSO
                                        k0scSrc0sc
54 /* RTC external clock configuration. */
55 #define RTC_XTAL_FREQ
56 #define RTC_SC2P_ENABLE_CONFIG
                                       false
57 #define RTC_SC4P_ENABLE_CONFIG
                                       false
58 #define RTC_SC8P_ENABLE_CONFIG
                                       false
59 #define RTC_SC16P_ENABLE_CONFIG
                                       false
60 #define RTC_OSC_ENABLE_CONFIG
                                       false
61 #define RTC_CLK_OUTPUT_ENABLE_CONFIG false
63 /* RTC_CLKIN PTC1 */
64 #define RTC_CLKIN_PORT
                                       PORTC
65 #define RTC_CLKIN_PIN
                                        1 U
66 #define RTC_CLKIN_PINMUX
                                       kPortMuxAsGpio
69 #define CLOCK_VLPR
                                       1U /* very low power run mode */
70 #define CLOCK_RUN
                                        2U /* run mode */
72 #ifndef CLOCK_INIT_CONFIG
73 #define CLOCK_INIT_CONFIG CLOCK_RUN
74 #endif
77 /* Configuration for enter VLPR mode, Core clock = 4MHz */
78 const clock_manager_user_config_t g_defaultClockConfigVlpr =
      .mcgConfig =
      {
81
                               = kMcgModeBLPI,
                                                   // Work in BLPI mode
          .mcg_mode
82
                                                    // MCGIRCLK enable
         .irclkEnable
                              = true,
                                                     // MCGIRCLK disable in STOP mode
          .irclkEnableInStop = false,
84
85
          .ircs
                               = kMcgIrcFast,
                                                    // Select IRC4M
          .fcrdiv
                               = OU,
                                                     // FCRDIV is 0
```

```
87
          .frdiv = OU,
                  = kMcgDcoRangeSelLow,
                                                   // Low frequency range
           .drs
           .dmx32 = kMcgDmx32Default,
                                                    // DCO has a default range of 25%
90
           .pl10EnableInFl1Mode = false,
                                                    // PLLO disable
92
           .pllOEnableInStop = false,
                                                    // PLLO disalbe in STOP mode
           .prdiv0
94
           .vdiv0
                            = OU,
       },
96
       .simConfig =
97
98
           .pllFllSel = kClockPllFllSelFll,
                                                 // PLLFLLSEL select FLL
99
           .er32kSrc = kClockEr32kSrcLpo,
                                                    // ERCLK32K selection, use LPO
100
101
           .outdiv1 = OU,
           .outdiv4 = 4U,
       },
103
104
       .oscerConfig =
105
                                                    // OSCERCLK enable
           .enable
                     = true,
106
           .enableInStop = false,
                                                     // OSCERCLK disable in STOP mode
108
109 };
111 /* Configuration for enter RUN mode, Core clock = 40 MHz */
^{113} * 24.5.1.1 Initializing the MCG
114 * KL25 Sub-Family Reference Manual, Rev. 3, September 2012
116 * Refer also to
* Table 24-18. MCG modes of operation
119 * On L-series devices the MCGFLLCLK frequency is limited to 48 MHz max
120 * The DCO is limited to the two lowest range settings (MCG_C4[DRST_DRS] must be set
      to either 0b00 or 0b01).
121 *
* FEE (FLL engaged external)
_{\rm 123} * fext / FLL_R must be in the range of 31.25 kHz to 39.0625 kHz
_{\rm 124} * FLL_R is the reference divider selected by the C1[FRDIV] bits
^{125} * F is the FLL factor selected by C4[DRST_DRS] and C4[DMX32] bits
* (fext / FLL_R) * F = (8 MHz / 256 ) * 1280 = 40 MHz
128 *
129 * */
```

```
130 const clock_manager_user_config_t g_defaultClockConfigRun =
      /* ----- multipurpose clock generator configurations ----- */
132
      .mcgConfig =
133
134
                             = kMcgModeFEE,
                                                // Work in FEE mode
135
          .mcg_mode
136
          /* ----- MCGIRCCLK settings ----- */
137
          .irclkEnable
                            = true,
                                                 // MCGIRCLK enable
                                                // MCGIRCLK disable in STOP mode
          .irclkEnableInStop = false,
139
                                                // Select IRC 32kHz
          .ircs
                             = kMcgIrcSlow,
140
          .fcrdiv
                             = OU,
                                                 // FCRDIV is 0
141
142
          /* ----- MCG FLL settings ---- */
143
144
          .frdiv = 0b011,
                                                // Divide Factor is 256 (EXT OSC 8
      MHz / 256 = 31.250 kHz)
                                                // The resulting frequency must be in
      the range 31.25 kHz to 39.0625 kHz
          .drs = kMcgDcoRangeSelMid,
                                                // frequency range
146
          .dmx32 = kMcgDmx32Default,
                                                // DCO has a default range of 25%
147
148
          /* ----- MCG PLL settings ----- */
149
          .pl10EnableInFl1Mode = false,
                                                // PLLO disable
          .pll0EnableInStop = false,
                                                // PLLO disabLe in STOP mode
151
          .prdiv0
                           = 0 \times 0 U,
152
          .vdiv0
                            = 0 \times 0 U,
153
      },
      /* ----- system integration module configurations -----**/
155
      .simConfig =
156
      {
          .pllFllSel = kClockPllFllSelFll,
                                                // PLLFLLSEL select PLL
158
          . er32kSrc = kClockEr32kSrcLpo ,
                                                // ERCLK32K selection, use LPO
159
          .outdiv1 = OU,
                                                // core/system clock, as well as the
160
      bus/flash clocks.
          .outdiv4 = 1U,
                                                // bus and flash clock and is in
161
      addition to the System clock divide ratio
162
      /* ----- system oscillator output configurations -----**/
      .oscerConfig =
164
      {
165
                                                 // OSCERCLK enable
          .enable
                     = true,
                                                 // OSCERCLK disable in STOP mode
          .enableInStop = false,
167
      }
168
169 };
```

```
170
172 /**
173 * Oscillator configuration
175 void mcg_initOscO(void)
176 {
       /* OSCO configuration */
177
       osc_user_config_t osc0Config =
178
179
                                  = OSCO_XTAL_FREQ,
           . frea
180
           . hgo
                                  = MCG_HGOO,
181
                                  = MCG_RANGEO,
182
           range
                                  = MCG_EREFSO,
           .erefs
183
           .enableCapacitor2p
                                  = OSCO_SC2P_ENABLE_CONFIG,
184
                                 = OSCO_SC4P_ENABLE_CONFIG,
           .enableCapacitor4p
185
           .enableCapacitor8p
                                  = OSCO_SC8P_ENABLE_CONFIG,
186
           .enableCapacitor16p = OSCO_SC16P_ENABLE_CONFIG,
187
       };
188
189
       /* oscillator initialization */
190
       CLOCK_SYS_OscInit(OSCO_INSTANCE, &oscOConfig);
191
192 }
193
194
195
197 * Function to initialize RTC external clock base on board configuration
199 void mcg_initRtcOsc(void)
200 {
201
202 #if RTC_XTAL_FREQ
203
       // If RTC_CLKIN is connected, need to set pin mux. Another way for
       // RTC clock is set RTC_OSC_ENABLE_CONFIG to use OSCO, please check
204
       // reference manual for details
205
       PORT_HAL_SetMuxMode(RTC_CLKIN_PORT, RTC_CLKIN_PIN, RTC_CLKIN_PINMUX);
206
207 #endif
208
209 #if ((OSCO_XTAL_FREQ != 32768U) && (RTC_OSC_ENABLE_CONFIG))
210 #error Set RTC_OSC_ENABLE_CONFIG will override OSCO configuration and OSCO must be 32k.
211 #endif
212
       rtc_osc_user_config_t rtcOscConfig =
```

```
{
214
                               = RTC_XTAL_FREQ,
                              = RTC_SC2P_ENABLE_CONFIG,
216
          .enableCapacitor2p
          .enableCapacitor4p
                               = RTC_SC4P_ENABLE_CONFIG,
217
          .enableCapacitor8p
                              = RTC_SC8P_ENABLE_CONFIG,
218
           .enableCapacitor16p = RTC_SC16P_ENABLE_CONFIG,
219
           .enableOsc
                               = RTC_OSC_ENABLE_CONFIG,
220
      };
221
       /* OSC RTC initialization */
223
       CLOCK_SYS_RtcOscInit(OU, &rtcOscConfig);
224
225 }
226
227
228
230 * System clock configuration
231 */
232 void mcg_initSystemClock(void)
233 {
       /* Set system clock configuration. */
234
       #if (CLOCK_INIT_CONFIG == CLOCK_VLPR)
235
          CLOCK_SYS_SetConfiguration(&g_defaultClockConfigVlpr);
237
          CLOCK_SYS_SetConfiguration(&g_defaultClockConfigRun);
238
       #endif
239
240 }
241
242
245 /* Method name: mcg_clockInit
246 /* Method description: main board clk configuration */
247 /* Input params:
                         n/a
248 /* Output params:
                         n/a
250 void mcg_clockInit(void)
251 {
       /* enable clock for PORTs */
252
       CLOCK_SYS_EnablePortClock(PORTA_IDX);
253
      CLOCK_SYS_EnablePortClock(PORTC_IDX);
      CLOCK_SYS_EnablePortClock(PORTE_IDX);
255
256
      /* set allowed power mode to allow all */
```

```
SMC_HAL_SetProtection(SMC, kAllowPowerModeAll);
258
259
       /* configure OSCO pin mux */
260
       PORT_HAL_SetMuxMode(EXTALO_PORT, EXTALO_PIN, EXTALO_PINMUX);
261
       PORT_HAL_SetMuxMode(XTALO_PORT, XTALO_PIN, XTALO_PINMUX);
262
263
       /* setup OSCO */
264
       mcg_initOscO();
265
266
       /* setup OSC RTC */
267
       mcg_initRtcOsc();
268
269
       /* setup system clock */
270
       mcg_initSystemClock();
271
272 }
```

### 7.5 ../Sources/PIT/pit hal.h

```
2 /* File name:
                   pit_hal.h
3 /* File description: Header file containing the functions/methods
                    interfaces for handling the Periodic Interruption */
                   timer module
5 /*
6 /* Author name:
                                                                */
                    ddello
7 /* Creation date:
                    10abr2016
                                                                */
8 /* Revision date:
                    10abr2016
                                                                */
11 #ifndef SOURCES_PIT_PIT_HAL_H_
12 #define SOURCES_PIT_PIT_HAL_H_
14 /**
* Enables Periodic Interruption Timer module.
* (With the stop on debug flag set to on)
17 */
18 void pit_enable(void);
21 * Start interruptions for given timer, unchained mode.
22 * Timer interruptions are masked.
^{24} * <code>Oparam usTimer_numb</code> The number for the desired timer (0,1)
25 * @param uiTimer_period The number of bus_clock cycles between interrupts
26 * Oparam fpInterrupt_handler Timer interrupt handler routine address pointer
28 void pit_start_timer_interrupt(unsigned short usTimer_numb, unsigned int
     uiTimer_period , void (*fpInterrupt_handler)(void));
31 * Stop interruptions for given timer, unchained mode.
* Operam usTimer_numb The number for the desired timer (0,1)
void pit_stop_timer_interrupt(unsigned short usTimer_numb);
37 /**
38 * Mark interruption as handled for the given timer, this should be called by timer
* interruption handlers once they are finished.
41 * Oparam usTimer_numb The number for the desired timer (0,1)
```

```
42 */
43 void pit_mark_interrupt_handled(unsigned short usTimer_numb);
44
45 /**
46 * Pit interruption handler. Checks what timer caused the interruption and call the
47 * correct timer interruption handler.
48 */
49 void PIT_IRQHandler(void);
50 #endif /* SOURCES_PIT_PIT_HAL_H_ */
```

### 7.6 ../Sources/PIT/pit hal.c

```
2 /* File name: pit_hal.c
                                                             */
3 /* File description: File containing the functions/methods
4 /*
                   for handling the Periodic Interruption
5 /*
                   Timer module
6 /* Author name:
                  ddello
                                                             */
7 /* Creation date:
                   10abr2016
                                                             */
8 /* Revision date: 13abr2016
10 //Careful when handling PIT DOC! Bit endianness is inverted in relation to GPIO doc
12 #include "pit_hal.h"
13 #include "KL25Z/es670_peripheral_board.h"
15 #define PIT_IRQ_NUMBER PIT_IRQn
18 *Default timer interruption handler. Does nothing.
20 static void _nop_handler(void){
PIT_TFLGO |= PIT_TFLG_TIF(0x1u);
PIT_TFLG1 |= PIT_TFLG_TIF(0x1u);
23 }
25 static void (*fpTimerOHandler)(void) = &_nop_handler;
26 static void (*fpTimer1Handler)(void) = &_nop_handler;
29 * Pit interruption handler. Checks what timer caused the interruption and call the
30 * correct timer interruption handler.
31 */
32 void PIT_IRQHandler(void){
if(PIT_TFLGO){
     (*fpTimerOHandler)();
  if(PIT_TFLG1){
     (*fpTimer1Handler)();
38
39 }
* Enables Periodic Interruption Timer module.
```

```
* (With the stop on debug flag set to on)
44 */
45 void pit_enable(void){
SIM_SCGC6 |= SIM_SCGC6_PIT_MASK;
  PIT_MCR &= "PIT_MCR_MDIS(0x1u);
   //Freeze in debug mode
   PIT_MCR |= PIT_MCR_FRZ(0x1u);
  NVIC_ClearPendingIRQ(PIT_IRQ_NUMBER);
     NVIC_EnableIRQ(PIT_IRQ_NUMBER);
52 }
53
* Start interruptions for given timer, unchained mode.
* Timer interruptions are masked.
* Oparam usTimer_numb The number for the desired timer (0,1)
* @param uiTimer_period The number of bus_clock cycles between interrupts
60 * Oparam fpInterrupt_handler Timer interrupt handler routine address pointer
62 void pit_start_timer_interrupt(unsigned short usTimer_numb, unsigned int
      uiTimer_period, void (*fpInterrupt_handler)(void)){
   if(!usTimer_numb){
63
     fpTimerOHandler = fpInterrupt_handler;
     PIT_LDVALO = PIT_LDVAL_TSV(uiTimer_period);
     PIT_TCTRLO &= "PIT_TCTRL_CHN(0x1u); /*Disable chain mode*/
     67
     PIT_TCTRL0 |= PIT_TCTRL_TEN(0x1u); /*Enable timer 0*/
   }else{
69
     fpTimer1Handler = fpInterrupt_handler;
70
     PIT_LDVAL1 = PIT_LDVAL_TSV(uiTimer_period);
     PIT_TCTRL1 &= ~PIT_TCTRL_CHN(0x1u); /*Disable chain mode*/
72
     PIT_TCTRL1 |= PIT_TCTRL_TIE(0x1u); /*Enable interrupts for timer 1*/
7.3
     PIT_TCTRL1 |= PIT_TCTRL_TEN(0x1u); /*Enable timer 1*/
75
  }
76 }
77
79 * Stop interruptions for given timer, unchained mode.
s1 * Oparam usTimer_numb The number for the desired timer (0,1)
83 void pit_stop_timer_interrupt(unsigned short usTimer_numb){
84 if(!usTimer_numb){
    PIT_TCTRLO &= "PIT_TCTRL_TIE(0x1u);
```

```
PIT_TCTRLO &= ~PIT_TCTRL_TEN(0x1u);
86
     PIT_TCTRL1 &= ~PIT_TCTRL_TIE(0x1u);
      PIT_TCTRL1 &= ~PIT_TCTRL_TEN(0x1u);
89
91 }
92
_{94} * Mark interruption as handled for the given timer, this should be called by timer
* interruption handlers once they are finished.
   * @param usTimer_numb The number for the desired timer (0,1)
97
99 void pit_mark_interrupt_handled(unsigned short usTimer_numb){
100
   if(!usTimer_numb){
     PIT_TFLGO |= PIT_TFLG_TIF(0x1u);
101
    } else {
102
    PIT_TFLG1 |= PIT_TFLG_TIF(0x1u);
103
104
105 }
```

### 7.7 ../Sources/SevenSeg/sevenseg hal.h

```
2 /* File name: sevenseg_hal.h
3 /* File description: Header file containing the functions/methods
                                                            */
                  interfaces for handling SEVEN SEGMENT DISPLAY
5 /*
                   from the peripheral board
                                                            */
6 /* Author name:
                  ddello
                                                            */
                                                            */
7 /* Creation date: 18 mar 2016
8 /* Revision date:
                  13abr2016
11 #ifndef SOURCES_SEVEN_SEGMENT_HAL_H_
12 #define SOURCES_SEVEN_SEGMENT_HAL_H_
14 #include "KL25Z/es670_peripheral_board.h"
16 #define MAX_SEGMENT_NUMBER 8
17 #define MAX_DISP_NUMBER 4
20 typedef enum
21 {
     SEG_A = SEGA_PIN,
    SEG_B = SEGB_PIN,
     SEG_C = SEGC_PIN,
24
    SEG_D = SEGD_PIN,
    SEG_E = SEGE_PIN,
    SEG_F = SEGF_PIN,
    SEG_G = SEGG_PIN,
     SEG_DP = SEGDP_PIN,
  SEG\_END = -1
31 } seven_segment_seg_type_e;
33 typedef enum
34 {
     DISP_1 = SEG_DISP1_PIN,
  DISP_2 = SEG_DISP2_PIN,
37 DISP_3 = SEG_DISP3_PIN,
38 DISP_4 = SEG_DISP4_PIN,
39 } seven_segment_disp_type_e;
41 /**
42 * Initialize the seven segment display
```

```
43 */
44 void sevenseg_init(void);
46 /**
47 * Sets only the selected segments as high. Setting the others as low
48 * Oparam epDet_segments = Array with the segments that should be set as on (Last
      element should be SEG_END)
50 void sevenseg_setSegs(seven_segment_seg_type_e* epSet_segments);
52 /**
* Shows the value written in the segment pins to the
_{54} * given display after clearing the others
* Oparam eDisplay the display to initialize.
57 void sevenseg_setDisp(seven_segment_disp_type_e eDisplay);
59 /**
60 * Shows the passed value in hexadecimal format in the seven segment display.
* Cparam wiHex the value to be printed
63 void sevenseg_printHex(unsigned int uiHex);
65 /**
66 * Shows the passed value in decimal format in the seven segment display.
* Cparam uiDec the value to be printed
69 void sevenseg_printDec(unsigned int uiDec);
70
71 /**
_{72} * Converts the less significative decimal digit of the argument into it's seven
73 * segment display configuration
74 * Oparam usDec the value to be converted (-1 if none should be displayed)
75 * Oparam epRet address for results (should be a allocated array of minimal 9 elements)
76 *
77 * @return epRet
79 seven_segment_seg_type_e* sevenseg_dec2segArray(unsigned short usDec,
      seven_segment_seg_type_e* epRet);
80
81 /**
82 * Converts the less significative hexadecimal digit of the argument into it's seven
83 * segment display configuration
84 * Operam usHex the value to be converted (-1 if none should be displayed)
```

### 7.8 ../Sources/SevenSeg/sevenseg hal.c

```
2 /* File name: sevenseg_hal.c
3 /* File description: File containing the functions/methods
                                                                 */
4 /*
                    for handling SEVEN SEGMENT DISPLAY
5 /*
                     from the peripheral board
                                                                 */
6 /* Author name:
                                                                 */
                    ddello
7 /* Creation date:
                     18mar2016
                                                                 */
8 /* Revision date:
                    13abr2016
11 #include "GPIO/gpio_hal.h"
12 #include "sevenseg_hal.h"
13 #include "math.h"
14 #include "KL25Z/es670_peripheral_board.h"
15 #include "PIT/pit_hal.h"
17 #define SEV_SEG_SEGMENT_MASK GPIO_HIGH << SEGA_PIN | GPIO_HIGH << SEGB_PIN | GPIO_HIGH
     << SEGC_PIN | GPIO_HIGH << SEGD_PIN | GPIO_HIGH << SEGE_PIN | GPIO_HIGH << SEGF_PIN</pre>
     | GPIO_HIGH << SEGG_PIN | GPIO_HIGH << SEGDP_PIN
18 #define SEV_SEG_DISP_MASK GPIO_HIGH << SEG_DISP1_PIN | GPIO_HIGH << SEG_DISP2_PIN |
     GPIO_HIGH << SEG_DISP3_PIN | GPIO_HIGH << SEG_DISP4_PIN
20 #define SEVEN_SEG_PIT_PERIOD 0x0000F423 /* 62500 cycles = 3.125ms (20MHz)*/
22 static unsigned short usIsHex = 0;
23 static unsigned int uiPrintVal = -1;
26 * Interrupt handler for updating in display configuration
28 void _sevenseg_interrupt_handler(void){
   static seven_segment_disp_type_e epDisplays[] = {DISP_1, DISP_2, DISP_3, DISP_4};
  static seven_segment_seg_type_e epSeg_array[MAX_SEGMENT_NUMBER+1];
   static volatile unsigned short usCur_disp = 0;
   if (usIsHex) {
32
     sevenseg_hex2segArray(uiPrintVal/pow(16,MAX_DISP_NUMBER-1-usCur_disp),
     epSeg_array);
   }else{
34
     sevenseg_dec2segArray(uiPrintVal/pow(10,MAX_DISP_NUMBER-1-usCur_disp),
     epSeg_array);
   }
36
   sevenseg_setSegs(epSeg_array);
```

```
sevenseg_setDisp(epDisplays[usCur_disp]);
38
    usCur_disp = (usCur_disp+1) %MAX_DISP_NUMBER;
    pit_mark_interrupt_handled(SEV_SEG_PIT_TIMER_NUMB);
41 }
42
43 /**
44 * Initialize the seven segment display
46 void sevenseg_init(void){
    GPIO_UNGATE_PORT(SEV_SEG_PORT_ID);
48
    // Init the Seven Segment segment control pins as OUTPUT
49
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGA_PIN, GPIO_OUTPUT);
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGB_PIN, GPIO_OUTPUT);
51
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGC_PIN, GPIO_OUTPUT);
52
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGD_PIN, GPIO_OUTPUT);
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGE_PIN, GPIO_OUTPUT);
54
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGF_PIN, GPIO_OUTPUT);
55
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGG_PIN, GPIO_OUTPUT);
56
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEGDP_PIN, GPIO_OUTPUT);
    // Init the Seven Segment segment display pins as OUTPUT
58
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEG_DISP1_PIN, GPIO_OUTPUT);
59
    GPIO_INIT_PIN(SEV_SEG_PORT_ID , SEG_DISP2_PIN , GPIO_OUTPUT);
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEG_DISP3_PIN, GPIO_OUTPUT);
61
    GPIO_INIT_PIN(SEV_SEG_PORT_ID, SEG_DISP4_PIN, GPIO_OUTPUT);
62
63
    //Init pit interrupts
    pit_enable();
65
    //Init timer 0
    pit_start_timer_interrupt(SEV_SEG_PIT_TIMER_NUMB, SEVEN_SEG_PIT_PERIOD,
      &_sevenseg_interrupt_handler);
68 }
69
72 * Sets only the selected segments as high. Setting the others as low
73 * @param epDet_segments = Array with the segments that should be set as on (Last
      element should be SEG_END)
75 void sevenseg_setSegs(seven_segment_seg_type_e* epSet_segments){
  //Clear all segments.
    GPIO_WRITE_MASK(SEV_SEG_PORT_ID , SEV_SEG_SEGMENT_MASK , GPIO_LOW);
    //Set the selected segments to high
78
    for(unsigned short usCounter = 0; epSet_segments[usCounter] != SEG_END; usCounter++){
```

```
GPIO_WRITE_PIN(SEV_SEG_PORT_ID, epSet_segments[usCounter], GPIO_HIGH);
     }
82 }
83
84 /**
85 * Shows the value written in the segment pins to the
86 * given display after clearing the others
87 * Cparam eDisplay the display to initialize.
89 void sevenseg_setDisp(seven_segment_disp_type_e eDisplay) {
90 //Clear all displays
    GPIO_WRITE_MASK(SEV_SEG_PORT_ID , SEV_SEG_DISP_MASK , GPIO_LOW);
92 //Activate the selected display
GPIO_WRITE_PIN(SEV_SEG_PORT_ID, eDisplay, GPIO_HIGH);
94 }
96 /**
97 * Shows the passed value in hexadecimal format in the seven segment display.
98 * Oparam uiHex the value to be printed
void sevenseg_printHex(unsigned int uiHex){
   usIsHex = 1;
    uiPrintVal = uiHex;
103 }
104
105 /**
_{106} * Shows the passed value in decimal format in the seven segment display.
* Cparam uiDec the value to be printed
void sevenseg_printDec(unsigned int uiDec){
usIsHex = 0;
     uiPrintVal = uiDec;
111
112 }
113
114 /**
* Converts the less significative decimal digit of the argument into it's seven
116 * segment display configuration
117 * Operam usDec the value to be converted (-1 if none should be displayed)
118 * @param epRet address for results (should be a allocated array of minimal 9 elements)
119 *
120 * @return epRet
121 */
122 seven_segment_seg_type_e* sevenseg_dec2segArray(unsigned short usDec,
       seven_segment_seg_type_e* epRet){
```

```
if(usDec < 0){</pre>
123
       epRet[0] = SEG_END;
124
       return epRet;
125
126
     epRet[0] = SEG_A;
127
     epRet[1] = SEG_B;
128
     epRet[2] = SEG_C;
129
     epRet[3] = SEG_D;
130
     epRet[4] = SEG_E;
131
     epRet[5] = SEG_F;
132
     epRet[6] = SEG_G;
133
     epRet[7] = SEG_END;
134
     switch(usDec%10){
135
     case 0:
136
137
       //{SEG_A,SEG_B,SEG_C,SEG_D,SEG_G,SEG_E,SEG_F,SEG_END};
       epRet[7] = SEG_END;
138
       break;
139
140
     case 1:
       //{SEG_B,SEG_C,SEG_END};
141
       epRet[0] = SEG_B;
142
       epRet[1] = SEG_C;
143
       epRet[2] = SEG_END;
144
       break;
     case 2:
146
       //{SEG_A,SEG_B,SEG_G,SEG_D,SEG_E,SEG_END};
147
       epRet[2] = SEG_G;
148
       epRet[5] = SEG_END;
149
       break;
150
     case 3:
151
       //{SEG_A,SEG_B,SEG_C,SEG_D,SEG_G,SEG_END}
       epRet[4] = SEG_G;
153
       epRet[5] = SEG_END;
154
       break;
155
     case 4:
       //{SEG_G, SEG_B,SEG_C,SEG_F, SEG_END}
157
       epRet[0] = SEG_G;
158
       epRet[3] = SEG_F;
159
       epRet[4] = SEG_END;
160
       break:
161
     case 5:
162
       //{SEG_A,SEG_G,SEG_C,SEG_D,SEG_F,SEG_END}
       epRet[1] = SEG_G;
164
       epRet[4] = SEG_F;
165
       epRet[5] = SEG_END;
166
```

```
break;
167
     case 6:
       //{SEG_A,SEG_G,SEG_C,SEG_D,SEG_E,SEG_F,SEG_END}
169
       epRet[1] = SEG_G;
170
       epRet[6] = SEG_END;
171
       break;
172
     case 7:
173
       //{SEG_A,SEG_B,SEG_C,SEG_END}
174
       epRet[3] = SEG_END;
       break;
176
177
     case 8:
       //{SEG_A,SEG_B,SEG_C,SEG_D,SEG_E,SEG_F,SEG_G,SEG_END}
       break;
179
     case 9:
180
181
       //SEG_A,SEG_B,SEG_C,SEG_F,SEG_G,SEG_END}
       epRet[3] = SEG_F;
182
       epRet[4] = SEG_G;
183
       epRet[5] = SEG_END;
184
       break;
185
     }
186
     return epRet;
187
188 }
189
191 * Converts the less significative hexadecimal digit of the argument into it's seven
   * segment display configuration
   * Oparam usHex the value to be converted (-1 if none should be displayed)
   * Cparam epRet address for results (should be a allocated array of minimal 9 elements)
194
195
196 * @return epRet
198 seven_segment_seg_type_e* sevenseg_hex2segArray(unsigned short usHex,
       seven_segment_seg_type_e* epRet){
     if(usHex < 0){</pre>
       epRet[0] = SEG_END;
200
       return epRet;
201
202
     epRet[0] = SEG_A;
203
     epRet[1] = SEG_B;
204
     epRet[2] = SEG_C;
205
     epRet[3] = SEG_D;
     epRet[4] = SEG_E;
207
     epRet[5] = SEG_F;
208
     epRet[6] = SEG_G;
```

```
epRet[7] = SEG_END;
210
211
     switch(usHex%16){
       case 0:
212
       case 1:
213
214
       case 2:
       case 3:
215
       case 4:
216
       case 5:
217
       case 6:
       case 7:
219
       case 8:
220
221
       case 9:
         return sevenseg_dec2segArray(usHex%16, epRet);
222
        break:
223
224
       case 10: //A
         //{SEG_A,SEG_B,SEG_C,SEG_G,SEG_E,SEG_F,SEG_END}
225
         epRet[3] = SEG_G;
226
         epRet[6] = SEG_END;
227
         break;
228
       case 11: //B (b)
229
         //{SEG_G,SEG_F,SEG_C,SEG_D,SEG_E,SEG_END}
230
         epRet[0] = SEG_G;
231
         epRet[1] = SEG_F;
         epRet[5] = SEG_END;
233
         break;
234
235
       case 12: //C
         //{SEG_A,SEG_E,SEG_F,SEG_D,SEG_END}
         epRet[1] = SEG_E;
237
         epRet[2] = SEG_F;
238
239
         epRet[4] = SEG_END;
         break;
240
       case 13: //D (d)
241
         //{SEG_G,SEG_B,SEG_C,SEG_D,SEG_E,SEG_END}
242
         epRet[0] = SEG_G;
         epRet[5] = SEG_END;
244
         break;
245
       case 14: //E
246
         //{SEG_A,SEG_G,SEG_F,SEG_D,SEG_E,SEG_END}
247
         epRet[1] = SEG_G;
248
         epRet[2] = SEG_F;
249
         epRet[5] = SEG_END;
250
         break;
251
       case 15: //F
252
         //{SEG_A,SEG_E,SEG_F,SEG_G,SEG_END}
```

#### 7.9 ../Sources/GPIO/gpio hal.h

```
2 /* File name: gpio_hal.h
3 /* File description: This file has a couple of useful macros to
                                                              */
                    control and init the GPIO pins from the KLM25Z
5 /* Author name:
                   ddello
6 /* Creation date: 01abr2016
                                                              */
7 /* Revision date:
                    13abr2016
                                                              */
10 #ifndef SOURCES_GPIO_GPIO_HAL_H_
11 #define SOURCES_GPIO_GPIO_HAL_H_
# include "KL25Z/es670_peripheral_board.h"
15 /* GPIO input / output */
16 #define GPIO_INPUT
                                 0 x 0 0 U
17 #define GPIO_OUTPUT
                                 0 x 0 1 U
19 #define GPIO_MUX_ALT
                                 0 x 0 1 u
21 #define GPIO_HIGH 1
22 #define GPIO_LOW
25 * Ungates the clock for a gpio port
26 * Oparam PORT_ID the GPIO port id(A,B)
28 #define GPIO_UNGATE_PORT(PORT_ID)\
  _GPIO_UNGATE_PORT(PORT_ID)
_{31} //Wrapper macro above is needed for argument expansion when using concatenation
32 #define _GPIO_UNGATE_PORT(PORT_ID)\
33 /* un-gate port clock*/\
    SIM_SCGC5 |= SIM_SCGC5_PORT ## PORT_ID (CGC_CLOCK_ENABLED)
* inits a pin as GPIO in the given direction
* Oparam PORT_ID the GPIO port id(A,B)
* @param PIN_NUM pin number in port
* @param DIR pin direction (GPIO_HIGH, GPIO_LOW)
42 #define GPIO_INIT_PIN(PORT_ID, PIN_NUM, DIR)\
```

```
_GPIO_INIT_PIN(PORT_ID, PIN_NUM, DIR)
43
45 //Wrapper macro above is needed for argument expansion when using concatenation
46 #define _GPIO_INIT_PIN(PORT_ID, PIN_NUM, DIR)\
      /* set pin as gpio */\
      PORT ## PORT_ID ## _PCR ## PIN_NUM = PORT_PCR_MUX(GPIO_MUX_ALT);
      /* Set pin direction */\
49
     if(DIR == GPIO_OUTPUT){\
50
        GPIO ## PORT_ID ## _PDDR |= GPIO_PDDR_PDD(0x01 << PIN_NUM);\</pre>
52
        GPIO ## PORT_ID ## _PDDR &= ~GPIO_PDDR_PDD(0x01 << PIN_NUM);\</pre>
53
54
56
* Writes a pin with the given value
* Oparam PORT_ID the GPIO port id(A,B)
* Coaram PIN_NUM pin number in port
* Cparam VAL pin value (GPIO_HIGH, GPIO_LOW)
63 #define GPIO_WRITE_PIN(PORT_ID, PIN_NUM, VAL)
      _GPIO_WRITE_PIN(PORT_ID, PIN_NUM, VAL)
66 //Wrapper macro above is needed for argument expansion when using concatenation
67 #define _GPIO_WRITE_PIN(PORT_ID, PIN_NUM, VAL)\
    if(VAL == GPIO_HIGH){\
      GPIO ## PORT_ID ## _PSOR = GPIO_PSOR_PTSO( (0x01U << PIN_NUM) );\
    }else{\
      GPIO ## PORT_ID ## _PCOR = GPIO_PCOR_PTCO( (0x01U << PIN_NUM) );\
71
73
74 /**
75 * Writes the given value to the pins given in the MASK
* Oparam PORT_ID the GPIO port id(A,B)
77 * Operam MASK 31 bit Mask with 1 in the bits corresponding to the pins of interest.
78 * Oparam VAL pins value (GPIO_HIGH, GPIO_LOW)
80 #define GPIO_WRITE_MASK(PORT_ID, MASK, VAL)\
      _GPIO_WRITE_MASK(PORT_ID, MASK, VAL)
83 #define _GPIO_WRITE_MASK(PORT_ID, MASK, VAL)\
84 if(VAL == GPIO_HIGH){\
      GPIO ## PORT_ID ## _PSOR = GPIO_PSOR_PTSO(MASK);\
  }else{\
```

```
87
      GPIO ## PORT_ID ## _PCOR = GPIO_PCOR_PTCO(MASK);\
89
90
91 /**
_{92} * Reads the status of a GPIO PIN
93 * @param PORT_ID the GPIO port id(A,B)
94 * @param PIN_NUM pin number in port
* @param VAL pin value (GPIO_HIGH, GPIO_LOW)
97 #define GPIO_READ_PIN(PORT_ID, PIN_NUM)\
98
       _GPIO_READ_PIN(PORT_ID, PIN_NUM)
_{100} //Wrapper macro above is needed for argument expansion when using concatenation
101 #define _GPIO_READ_PIN(PORT_ID, PIN_NUM) \
      ((GPIO ## PORD_ID ## _PDIR & (0x01u << PIN_NUM)) >> PIN_NUM))
102
104 #endif /* SOURCES_GPIO_GPIO_HAL_H_ */
```

## 7.10 .../Sources/Main/es670.c

```
1 #include "KL25Z/es670_peripheral_board.h"
2 #include "LedSwi/ledswi_hal.h"
3 #include "Mcg/mcg_hal.h"
4 #include "Buzzer/buzzer_hal.h"
5 #include "SevenSeg/sevenseg_hal.h"
6 #include "PIT/pit_hal.h"
7 #include "Util/util.h"
10 int main(void)
    mcg_clockInit();
    ledswi_initLedSwitch(1,3);
13
  sevenseg_init();
14
    sevenseg_printHex(0xABCDu);
    buzzer_initPeriodic(0xB18Eu); //440Hz (Base 20MHz)
    ledswi_setLed(4);
17
    while(1){
      if(ledswi_getSwitchStatus(3) == SWITCH_OFF || ledswi_getSwitchStatus(2) ==
      SWITCH_OFF || ledswi_getSwitchStatus(1) == SWITCH_OFF){
        ledswi_setLed(0x4);
20
      } else{
        ledswi_clearLed(0x4);
23
      /* Never leave main */
      return 0;
26
27 }
```

# $7.11 \quad ../Sources/Buzzer/buzzer\_hal.c$

```
2 /* File name: buzzer_hal.c
3 /* File description: File dedicated to the hardware abstraction layer*/
                   related buzzer from the peripheral board
5 /* Author name:
                  dloubach
                                                           */
6 /* Creation date: 12jan2016
                                                           */
7 /* Revision date:
                   13abr2016
                                                           */
10 #include "GPIO/gpio_hal.h"
11 #include "buzzer_hal.h"
12 #include "KL25Z/es670_peripheral_board.h"
13 #include "PIT/pit_hal.h"
14
16 * Initialize the buzzer device
17 */
18 void buzzer_init(void)
   GPIO_UNGATE_PORT(BUZZER_PORT_ID);
   GPIO_INIT_PIN(BUZZER_PORT_ID, BUZZER_PIN, GPIO_OUTPUT);
pit_enable();
23 }
24
28 * Clear the buzzer
30 void buzzer_clearBuzz(void)
     GPIO_WRITE_PIN(BUZZER_PORT_ID, BUZZER_PIN, GPIO_LOW);
33 }
34
38 * Set the buzzer
40 void buzzer_setBuzz(void)
GPIO_WRITE_PIN(BUZZER_PORT_ID, BUZZER_PIN, GPIO_HIGH);
```

```
43 }
45 /**
46 * Handler for buzzer interruptions
48 void _buzzer_interrupt_handler(void){
   static volatile unsigned short usBusOn = 0;
   if(!usBusOn){
50
      buzzer_setBuzz();
51
   }else{
52
     buzzer_clearBuzz();
53
54
  usBusOn = !usBusOn;
pit_mark_interrupt_handled(BUZZER_PIT_TIMER_NUMB);
57 }
59 /**
80 * Starts the buzzer with the specified period
^{62} * Operam uiPeriod The period of the buzzer signal, in clock cycles (40MHz)
64 void buzzer_initPeriodic(unsigned int uiPeriod){
65 //Init timer 1
pit_start_timer_interrupt(BUZZER_PIT_TIMER_NUMB, uiPeriod/2,
      &_buzzer_interrupt_handler);
67 }
69 /**
70 * Stops any periodic buzzer signal
71 */
72 void buzzer_stopPeriodic(void){
pit_stop_timer_interrupt(BUZZER_PIT_TIMER_NUMB);
74 }
```

## 7.12 ../Sources/Buzzer/buzzer hal.h

```
2 /* File name: buzzer_hal.h
                                                            */
3 /* File description: Header file containing the functions/methods
                                                            */
                  interfaces for handling BUZZER from the
5 /*
                  peripheral board
                                                            */
6 /* Author name:
                  dloubach
                                                            */
7 /* Creation date: 12jan2016
                                                            */
8 /* Revision date: 13abr2016
                                                            */
11 #ifndef SOURCES_BUZZER_HAL_H_
12 #define SOURCES_BUZZER_HAL_H_
14 /**
15 * Initialize the buzzer device
17 void buzzer_init(void);
20 /**
21 * Clear the buzzer
23 void buzzer_clearBuzz(void);
26 /**
27 * Set the buzzer
29 void buzzer_setBuzz(void);
31 /**
32 * Starts the buzzer with the specified period
^{34} * Oparam uiPeriod The period of the buzzer signal, in clock cycles (40MHz)
36 void buzzer_initPeriodic(unsigned int uiPeriod);
* Stops any periodic buzzer signal
41 void buzzer_stopPeriodic(void);
```

```
43
44 #endif /* SOURCES_BUZZER_HAL_H_ */
```

#### 7.13 ../Sources/KL25Z/es670 peripheral board.h

```
2 /* File name: es670_peripheral_board.h
_3 /* File description: Header file containing the peripherals mapping */
                   of the peripheral board for the ES670 hardware*/
5 /* Author name:
                  dloubach
6 /* Creation date: 16 out 2015
                                                             */
7 /* Revision date:
                   25 fe v 2 0 1 6
                                                            */
10 #ifndef SOURCES_ES670_PERIPHERAL_BOARD_H_
11 #define SOURCES_ES670_PERIPHERAL_BOARD_H_
13 /* system includes */
14 #include <MKL25Z4.h>
16 /*
                 General uC definitions
                                                  */
18 /* Clock gate control */
19 #define CGC_CLOCK_DISABLED
                               0 x 0 0 U
20 #define CGC_CLOCK_ENABLED
                               0 x 0 1 U
22 /* GPIO DIRECTION */
23 #define GPIO_OUTPUT
                        0 x 0 1 U
25 /* Workaround for PORT_ID macro expansion to stop at port level*/
26 typedef int A;
27 typedef int B;
28 typedef int C;
29 typedef int D;
30 typedef int E;
                END OF General uC definitions */
                 BUZZER Definitions
36 #define BUZZER_PORT_BASE_PNT PORTD
                                                                   /*
     peripheral port base pointer */
37 #define BUZZER_GPIO_BASE_PNT PTD
    peripheral gpio base pointer */
38 #define BUZZER_PORT_ID D
                                                             /* peripheral
    port identifier*/
```

```
40 #define BUZZER_PIT_TIMER_NUMB 1
42 #define BUZZER_PIN
                                                                            /* buzzer
     pin */
43 #define BUZZER_DIR
                                    kGpioDigitalOutput
44 #define BUZZER_ALT
                                     0 x 0 1 u
            END OF BUZZER definitions
                                                        */
                    LED and SWITCH Definitions
48 /*
                                                                  */
49 #define LS_PORT_BASE_PNT PORTA
      peripheral port base pointer */
50 #define LS_PORT_ID
     peripheral port identifier*/
51 #define LS_GPIO_BASE_PNT
                                    PTA
     peripheral gpio base pointer */
53 /* THIS PIN CONFLICTS WITH PTA1 USED AS UARTO_RX IN THE OPENSDA SERIAL DEBUG PORT */
54 #define LS1_PIN
                                  1 U
                                                                            /*
     led/switch #1 pin */
                                     (GPIO_OUTPUT << LS1_PIN)
55 #define LS1_DIR_OUTPUT
56 #define LS1_DIR_INPUT
                                     (GPIO_OUTPUT << LS1_PIN)
                                                                             /* GPIO
57 #define LS1_ALT
                                     0 x 0 1 u
     alternative */
59 /* THIS PIN CONFLICTS WITH PTA2 USED AS UARTO_TX IN THE OPENSDA SERIAL DEBUG PORT */
60 #define LS2_PIN
                                     2 U
     led/switch #2 pin */
61 #define LS2_DIR_OUTPUT
                                     (GPIO_OUTPUT << LS2_PIN)
62 #define LS2_DIR_INPUT
                                     (GPIO_OUTPUT << LS2_PIN)
63 #define LS2_ALT
                                     LS1_ALT
65 #define LS3_PIN
                                     4 U
                                                                             /*
     led/switch #3 pin */
66 #define LS3_DIR_OUTPUT
                                    (GPIO_OUTPUT << LS3_PIN)
67 #define LS3_DIR_INPUT
                                     (GPIO_OUTPUT << LS3_PIN)
68 #define LS3_ALT
                                     LS1_ALT
70 #define LS4_PIN
                                                                             /*
      led/switch #4 pin */
71 #define LS4_DIR_OUTPUT
                                    (GPIO_OUTPUT << LS4_PIN)
72 #define LS4_DIR_INPUT
                                     (GPIO_OUTPUT << LS4_PIN)
73 #define LS4_ALT
                                     LS1_ALT
```

```
75 /*
           END OF LED and SWITCH definitions
                                                       */
77 /*
                   SEVEN SEGMENT DISPLAY Definitions
78 #define SEV_SEG_PORT_BASE_PNT PORTC
      peripheral port base pointer */
79 #define SEV_SEG_PORT_ID
                          C
      peripheral port identifier*/
80 #define SEV_SEG_GPIO_BASE_PNT PTC
      peripheral gpio base pointer */
82 #define SEV_SEG_PIT_TIMER_NUMB 0
                                                      /* timer number for seven seg
     PIT */
84 #define SEGA_PIN
                                    0
                                                                           /* Segment
     A * /
85 #define SEGA_DIR_OUTPUT
                                    (GPIO_OUTPUT << SEGA_PIN)
86 #define SEGA_ALT
                                                                            /* GPIO
                                     0 x 0 1 u
      alternative */
88 #define SEGB_PIN
                                    (GPIO_OUTPUT << SEGB_PIN)
89 #define SEGB_DIR_OUTPUT
90 #define SEGB_ALT
                                    SEGA_ALT
92 #define SEGC_PIN
93 #define SEGC_DIR_OUTPUT
                                    (GPIO_OUTPUT << SEGC_PIN)
94 #define SEGC_ALT
                                     SEGA_ALT
96 #define SEGD_PIN
                                    (GPIO_OUTPUT << SEGD_PIN)
97 #define SEGD_DIR_OUTPUT
98 #define SEGD_ALT
                                     SEGA_ALT
100 #define SEGE_PIN
101 #define SEGE_DIR_OUTPUT
                                   (GPIO_OUTPUT << SEGE_PIN)
                                     SEGA_ALT
102 #define SEGE_ALT
103
104 #define SEGF_PIN
105 #define SEGF_DIR_OUTPUT
                                     (GPIO_OUTPUT << SEGF_PIN)
106 #define SEGF_ALT
                                    SEGA_ALT
108 #define SEGG_PIN
                                    (GPIO_OUTPUT << SEGG_PIN)
109 #define SEGG_DIR_OUTPUT
110 #define SEGG_ALT
                                     SEGA_ALT
111
112 #define SEGDP_PIN
                                     7
```

```
113 #define SEGDP_DIR_OUTPUT
                                          (GPIO_OUTPUT << SEGDP_PIN)
114 #define SEGDP_ALT
                                          SEGA_ALT
116 #define SEG_DISP1_PIN
117 #define SEG_DISP1_DIR_OUTPUT
                                         (GPIO_OUTPUT << SEG_DISP1_PIN)
118 #define SEG_DISP1_ALT
                                          SEGA_ALT
120 #define SEG_DISP2_PIN
                                          12
121 #define SEG_DISP2_DIR_OUTPUT
                                          (GPIO_OUTPUT << SEG_DISP2_PIN)
122 #define SEG_DISP2_ALT
                                          SEGA_ALT
123
124 #define SEG_DISP3_PIN
125 #define SEG_DISP3_DIR_OUTPUT
                                          (GPIO_OUTPUT << SEG_DISP3_PIN)
126 #define SEG_DISP3_ALT
                                          SEGA_ALT
128 #define SEG_DISP4_PIN
129 #define SEG_DISP4_DIR_OUTPUT
                                          (GPIO_OUTPUT << SEG_DISP4_PIN)
130 #define SEG_DISP4_ALT
                                          SEGA_ALT
                      END of SEVEN SEGMENT DISPLAY Definitions
                                                                                    */
132 /*
133
134
#endif /* SOURCES_ES670_PERIPHERAL_BOARD_H_ */
```

## 7.14 ../Sources/Util/util.h

```
2 /* File name: util.h
                                                             */
3 /* File description: Header file containing the function/methods
                                                             */
                   prototypes of util.c
5 /*
                   Those delays were tested under the following:
6 /*
                   core clock @ 40MHz
                                                             */
                   bus clock @ 20MHz
                                                             */
s /* Author name:
                  dloubach
                                                             */
9 /* Creation date: 09jan2015
                                                             */
10 /* Revision date: 13abr2016
                                                             */
13 #ifndef UTIL_H
14 #define UTIL_H
* generates ~ 088 micro sec
19 void util_genDelay088us(void);
21
* generates ~ 250 micro sec
void util_genDelay250us(void);
29 /* generates ~ 1 mili sec
31 void util_genDelay1ms(void);
* generates ~ 10 mili sec
37 void util_genDelay10ms(void);
40 #endif /* UTIL_H */
```

## 7.15 ../Sources/Util/util.c

```
2 /* File name: util.c
                                                            */
3 /* File description: This file has a couple of useful functions to
                                                            */
                   make programming more productive
5 /*
                                                            */
                   Remarks: The soft delays consider
6 /*
                                                            */
                   core clock @ 40MHz
                                                            */
                   bus clock @ 20MHz
8 /*
                                                            */
9 /*
                                                            */
10 /* Author name:
                  dloubach
                                                            */
11 /* Creation date: 09 jan2015
                                                            */
12 /* Revision date: 13abr2016
                                                            */
15 #include "util.h"
18 * generates ~ 088 micro sec
20 void util_genDelay088us(void)
21 {
     char i;
    for(i=0; i<120; i++)</pre>
24
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
31
        __asm("NOP");
32
        __asm("NOP");
        __asm("NOP");
34
        __asm("NOP");
35
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
        __asm("NOP");
     }
41 }
```

42

```
43
45 /**
* generates ~ 250 micro sec
48 void util_genDelay250us(void)
49 {
      char i;
50
      for(i=0; i<120; i++)</pre>
51
52
           __asm("NOP");
53
          __asm("NOP");
54
          __asm("NOP");
          __asm("NOP");
56
          __asm("NOP");
57
          __asm("NOP");
           __asm("NOP");
           __asm("NOP");
60
          __asm("NOP");
61
           __asm("NOP");
      }
63
      util_genDelay088us();
64
      util_genDelay088us();
66 }
67
70 /**
71 /* generates ~ 1 mili sec
73 void util_genDelay1ms(void)
74 {
      util_genDelay250us();
      util_genDelay250us();
      util_genDelay250us();
      util_genDelay250us();
78
79 }
81
* generates ~ 10 mili sec
85 void util_genDelay10ms(void)
```

```
util_genDelay1ms();
87
      util_genDelay1ms();
      util_genDelay1ms();
      util_genDelay1ms();
90
      util_genDelay1ms();
91
      util_genDelay1ms();
      util_genDelay1ms();
93
94
      util_genDelay1ms();
      util_genDelay1ms();
      util_genDelay1ms();
96
97 }
```