# Desenvolvimento em microcontroladores baseados em processadores ARM Cortex-M4 com TivaWare

Leandro Fabian Junior Callebe Soares Barbosa Orientador: Gustavo Weber Denardin

2016

# Lista de Figuras

| 2.1  | Diagrama de Blocos - Processador Cortex-M3 e Cortex-M4 [1]                   | 6  |
|------|------------------------------------------------------------------------------|----|
| 2.2  | Modos de Operação [1]                                                        | 7  |
| 2.3  | Banco de registradores internos [1]                                          |    |
| 3.1  | Diagrama de Blocos - TM4C1294NCPDT [1] $\ \ldots \ \ldots \ \ldots \ \ldots$ | 12 |
| 4.1  | Criando um novo projeto                                                      | 13 |
| 4.2  | Configurando o projeto                                                       |    |
| 4.3  |                                                                              | 15 |
| 4.4  | •                                                                            | 15 |
| 4.5  | Configurando endereço de início do Linker do GCC                             | 16 |
|      |                                                                              |    |
| 5.1  | Janela de importação de arquivos                                             |    |
| 5.3  | · · · · · · · · · · · · · · · · · · ·                                        | 18 |
| 5.2  | Incluindo diretórios para compilação                                         | 19 |
| 8.1  | Protocolo de envio na comunicação UART                                       | 32 |
| 8.2  | Sinal de Transmissão UART no Tiva TM4C1294NCPDT                              |    |
|      |                                                                              |    |
| 9.1  | Padrão de Comunicação SPI                                                    |    |
| 9.2  | Diagrama de Comunicação SPI - Vários Escravos                                | 36 |
| 10.1 | Conversor A/D tipo Aproximação Sucessiva                                     | 38 |
| 11.1 | PWM modo Down [1]                                                            | 40 |
|      | PWM modo Down [1]                                                            |    |
|      | PWM modo Down [1]                                                            |    |
|      | PWM modo Down [1]                                                            |    |
|      |                                                                              |    |

# Lista de Tabelas

| 3.1  | Características Básicas - TM4C1294NCPDT [1]                                           | 1  |
|------|---------------------------------------------------------------------------------------|----|
| 8.1  | Canais do UART - Tiva TM4C1294NCPDT [1]                                               | 33 |
| 9.1  | Canais do SPI - Tiva TM4C1294NCPDT [1] $\ \ldots \ \ldots \ \ldots \ \ldots$          | 36 |
| 10.1 | Canais de Entrada ADC - Tiva TM4C1294NCPDT [1] $aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa$ | 39 |
| 12.1 | Características Básicas - TM4C1294NCPDT [1]                                           | 4: |

# Índice

| 1  | Introdução                                                                                           | 5                          |
|----|------------------------------------------------------------------------------------------------------|----------------------------|
| 2  | Conhecendo o Processador ARM  2.1 Características ARM                                                | 6<br>6<br>7<br>8           |
| 3  | Conhecendo a plataforma de trabalho                                                                  | 10                         |
| 4  | Iniciando um projeto no Code Composer                                                                | 13                         |
| 5  | Biblioteca TivaWare  5.1 Incluindo a TivaWare ao projeto                                             | 17<br>17<br>17<br>18<br>19 |
| 6  | Sistema de Clock 6.1 Fontes de clock 6.2 Circuito de verificação do MOSC 6.3 Na TivaWare 6.4 Exemplo | 20<br>20<br>21<br>21<br>25 |
| 7  | ( )                                                                                                  |                            |
| 8  | UART  8.1 Padrão da Comunicação                                                                      |                            |
| 9  | SPI 9.1 Padrão da Comunicação                                                                        | 35<br>35<br>36<br>37       |
| 10 | 0 ADC         10.1 ADC de Aproximações Sucessivas                                                    |                            |

| ÍNDICE                                        | 4            |
|-----------------------------------------------|--------------|
| 11 PWM         11.1 Modos de Funcionamento    |              |
| 12 Exemplos de aplicação<br>12.1 Echo da UART | <b>43</b> 43 |

# Introdução

O material aqui contido busca iniciar o leitor no desenvolvimento em plataformas baseadas em processadores com arquitetura ARM.

O Hardware utilizado para os exemplos aqui apresentados é o Kit de avaliação da Texas Instruments Tiva $^{\rm TM}$ C Series TM4C1294.

### Conhecendo o Processador ARM

#### 2.1 Características ARM

O termo ARM se refere a Advanced RISC Machine, ou seja uma arquitetura que usa o conceito conhecido como RISC. RISC, Reduced Instruction Set Computer, é uma linha de arquitetura que possui um conjunto simples e pequeno de instruções que levam aproximadamente a mesma quantidade de tempo para serem executadas, permitindo que estes processadores tenham menos transístores do que aqueles projetados na arquitetura convencional. Logo essa abordagem reduz a liberação de calor, o consumo de energia e a quantidade de componentes em um processador.

A arquitetura dos processadores usados aqui, o Cortex-M3 e Cortex-M4, são ambos as implementações da arquitetura ARMv7-M. Existem diferentes tipos de arquitetura ARM para diferentes tipos de processadores, que ainda podem variar conforme são atualizadas ao longo dos anos. Os detalhes da arquitetura ARMv7-M estão documentados no Manual de Referência da Arquitetura ARMv7-M, disponível no site da ARM Limited.

O Cortex-M3 e Cortex-M4 são essencialmente idênticos em seus aspectos construtivos, de modo que o diagrama de blocos da figura 2.1 apresenta uma visão geral interna adequada tanto do processador Cortex –M4 quanto -M3.



Figura 2.1: Diagrama de Blocos - Processador Cortex-M3 e Cortex-M4 [1]

Na figura 2.1 notamos a presença de elementos no processador como: o controlador de vetores de interrupção, NVIC (Nested Vectored Interrupt Controller); o controlador de acionamento de interrupção, WIC (Wakeup Interrupt Controller); o temporizador SysTick;

a unidade de proteção de memória MPU (Memory Protection Unit); e uma unidade de ponto flutuante presente apenas no Cortex –M4. Existe ainda um sistema de debug dentro do processador para realizar depuração de software e um sistema interno de barramentos para transferência de dados entre o núcleo do processador, o sistema de debug e o MPU.

Os processadores da família Cortex M são de 32 bits, podendo também trabalhar com dados de 8 bits e 16 bits de forma bastante eficiente. Já os processadores Cortex-M3 e Cortex-M4, mesmo sendo da família Cortex M, podem realizar uma série de operações envolvendo dados de 64 bits. Estas operações podem ser realizadas através de um *piperline* de três estágios com uma arquitetura de barramento do tipo *Harvard* permitindo instruções simultâneas de busca e acesso de dados.

Uma das grandes vantagens dos processadores Cortex M é seu baixo consumo de energia. Em especial os processadores Cortex M3 e Cotex M4 podem executar instruções com taxa de 200mA/MHz com alimentação de 1,8V. Estes processadores possuem modos de suspensão que tornam possíveis desativar dispositivos de Clock para economizar energia, e um hardware adicional para despertar o processador dos modos de suspensão.

Devemos salientar aqui que estamos sempre nos referindo a apenas aos processadores, e que este é uma parte constituinte do microcontrolador. De modo que os demais componentes da placa são desenvolvidos pelos diferentes fabricantes. Assim existem vários tipos de microcontroladores com diferentes características de periféricos e recursos, porém a arquitetura empregada nos processadores é a mesma.

#### 2.2 Modos de operação ARM Cortex-M4

O processador Cortex-M4 possui dois estados de operação, como mostrado na figura 2.2, debug state e Thumb state. O debug state ocorre quando o processador é interrompido, por exemplo ao atingir um breakpoint, então a execução de instrução é interrompida. Já o Thumb state ocorre quando o código do programa está sendo executado. Diferente de outros processadores ARM, o Cortex-M não suporta instruções ARM.



Figura 2.2: Modos de Operação [1]

No *Thumb state* ainda existem dois modos de operação, que dizem respeito ao nível de privilégio no acesso ao processador. Ao executar uma rotina de tratamento de interrupção o processador entra em um nível de acesso privilegiado, caracterizando o *handler mode*. Durante a execução de uma aplicação normal o processador pode estar tanto em nível de acesso privilegiado quanto em nível menor, sendo chamado de *thread mode*. Isso é controlado por um registrador específico.

A aplicação pode alterar seu nível de acesso durante o *thread mode*, para um nível menos privilegiado. Porém, para aumentar seu nível de acesso deve haver um mecanismo de exceção/interrupção por parte do processador. Tais mecanismos de controle de nível de acesso garantem uma maior robustez para o sistema, controlando o acesso à regiões críticas de memória.

#### 2.3 Registradores internos

Para um controle melhor e um processamento de dados maior o Cortex-M4 possui registradores internos ao processador agrupados em um conjunto chamado de banco de registradores. Cada instrução enviada ao processador específica a operação a ser executada, os registradores fonte e se necessário os registradores de destino. A arquitetura ARM é baseada no modelo conhecido como load/store, ou seja, para processar um conteúdo que esteja na memória é preciso carregá-lo para um registrador interno e então processá-lo. Se necessário, é preciso armazená-lo de volta na memória.



Figura 2.3: Banco de registradores internos [1]

O banco de registradores do Cortex-M4 possui 16 registradores de 32 bits, como mostrado na Figura 2.3. Cada registrador possui seu propósito, como detalhado a seguir:

#### R0 - R12: Registradores de Propósito Geral

Devido ao número limitado no conjunto de instruções, muitas das de 16 bits somente acessam os registradores de R0 à R7, chamados de *registradores inferiores*. De R8 à R9, os *registradores altos*, podem ser usados com as instruções de 32 de bits e alguns com instruções 16 de bits. Os valores iniciais desses registradores são indefinidos.

#### R13: Ponteiro de Pilha (Stack Pointer, SP)

Usado para acessar a pilha de memória. Fisicamente há dois ponteiros de pilha, o principal (Main Stack Pointer, MSP) e o de processo (Process Stack Pointer, PSP)). O MSP é o ponteiro padrão, é selecionado após um reset do sistema ou quando o processador está em modo de exceção (Handler Mode). Seu valor inicial é o primeiro da memória na sequência de reset. Já o PSP é usado durante o Thread Mode, quando as tarefas da aplicação estão rodando, seu valor inicial é desconhecido.

Somente um dos ponteiros de pilha é visível durante a aplicação e os dois bits menos significativos de ambos são sempre nulos. Em aplicações que não fazem uso de um sistema operacional somente o MSP é usado.

#### R14: Registrador de Ligação (Link Register, LR)

Esse registrador armazena automaticamente o ponto em que uma rotina chama uma sub-rotina. Assim, ao fim da execução dessa sub-rotina, esse valor é carregado para o Contador de Programa e a execução continua de onde tinha anteriormente parado.

Se uma sub-rotina chamar outra sub-rotina, o valor nesse registrador será substituído e o ponto de retorno antigo se perderá, portanto é preciso que esse último valor seja salvo na pilha de memória.

Durante uma rotina de tratamento de exceção, o valor de LR é também sobrescrito mas por um valor de retorno de exceção, usado para disparar o retorno da exceção ao fim da rotina de tratamento.

#### R15: Contador de Programa (Program Counter, PC)

Marca o próximo endereço que deve ser executado na aplicação. Quando este registrador é lido, automaticamente seu valor decrementa de 4 (32 bits), apontando para o próximo endereço da execução. Já quando é feito uma operação de escrita, o programa pula para a posição apontada e passa a executar a aplicação a partir deste novo ponto.

O bit menos significativo do PC indica o tipo de instrução que está sendo executada, '0' para ARM e '1' para Thumb. Portanto no Cortex-M4, tal bit deve ser sempre '1' pois não são suportadas instruções ARM. Este fato deve ser lembrado quando é feita uma operação de escrita sobre o registrador.

# Conhecendo a plataforma de trabalho

O hardware utilizado aqui será o TIVA  $^{\intercal M}$  TM4C1294NCPDT, um kit de desenvolvimento da empresa Texas Instruments que possui um microcontrolador baseado no processador ARM Cortex-M4. A tabela 3.1 traz suas principais características.

| Características                           | Descrição                                                       |
|-------------------------------------------|-----------------------------------------------------------------|
| Núcleo                                    | ARM Cortex-M4F                                                  |
| Performance                               | Operação até 120-MHz; 150 DMIPS                                 |
|                                           | (Dhrystone MIPS) de performance                                 |
| Memória Flash                             | 1024 KB                                                         |
| SRAM                                      | 256 KB single-cycle System SRAM                                 |
| EEPROM                                    | 6KB                                                             |
| ROM                                       | ROM interna carregada com biblioteca                            |
|                                           | $\text{TivaWare}^{\text{TM}} \text{ C Series}$                  |
| Interface de Periféricos Externos (EPI)   | Interface dedicada de 8-/16-/32-                                |
|                                           | bits dedicados a periféricos e memoria                          |
| Verificação de Redundância                | Função Hash de 16-/32- bits, que suporta                        |
| Cíclica (CRC)                             | quatro formas de CRC                                            |
| Universal Asynchronous                    | 8 módulos UARTs                                                 |
| Receivers/Transmitter (UART)              |                                                                 |
| Quad Synchronous Serial                   | Quatro módulos de SSI com Bi- , Quad-                           |
| Interface (QSSI)                          | e suporte avançado de SSI                                       |
| Inter-Integrated Circuit $(I^2C)$         | $10 \text{ m\'odulos } I^2C \text{ com } 4 \text{ velocidades}$ |
|                                           | de transmissão                                                  |
| Controller Area Network (CAN)             | 2 controladores CAN 2.0 A/B                                     |
| Ethernet MAC                              | 10/100 Ethernet MAC                                             |
| Ethernet PHY                              | PHY com IEEE 1588 PTP                                           |
| Universal Serial Bus (USB)                | USB 2.0 OTG/Host/Device                                         |
|                                           | com ULPI interface e suporte a Link                             |
|                                           | Power Management (LPM)                                          |
| Micro Acesso Direto à Memória $(\mu DMA)$ | Controlador ARM© PrimeCell©                                     |
|                                           | 32-channel configurável $\mu DMA$                               |
| General-Purpose Timer (GPTM)              | 8 blocos 16/32-bit GPTM                                         |
| Watchdog Timer (WDT)                      | 2 Watchdog Timers                                               |
| Hibernation Module (HIB)                  | Low-power battery-backed                                        |
|                                           | Hibernation module                                              |
| General-Purpose Input/Output (GPIO)       | 15 physical GPIO blocks                                         |

| Pulse Width Modulator (PWM)        | 1 modulo PWM , com 4 geradores PWM    |
|------------------------------------|---------------------------------------|
|                                    | e um registador de controle,          |
|                                    | com um total de 8 saídas PWM.         |
| Quadrature Encoder Interface (QEI) | Um modulo QEI                         |
| Analog-to-Digital Converter (ADC)  | 2 modulos ADC de 12-bit               |
|                                    | taxa de 2 milhões de amostras/segundo |
| Controlador Comparador Analógico   | Três comparadores analógicos          |
|                                    | independentes                         |
| Comparador Digital                 | 16 comparadores digitais              |
| JTAG e Serial Wire Debug (SWD)     | 1 modulo JTAG com ARM SWD             |
|                                    | integrado                             |
| Encapsulamento                     | 128-pin TQFP                          |
| Temperatura de Operação            | $-40^{\circ}C$ até $105^{\circ}C$     |

Tabela 3.1: Características Básicas - TM4C1294NCPDT [1]

O TM4C1294NCPDT possui dois barramentos. Um desses é responsável pela conexão padrão entre o núcleo de processamento e os periféricos, o *Advanced Peripheral Bus* (APB). Já o outro, o *Advanced High-Performance Bus* (AHB), é um barramento especial que pode ser requisitado da maioria dos periféricos e, possui uma resposta muito mais rápida por ser exclusivo. O esquema dos barramentos é mostrado no diagrama de blocos da figura 3.1.



Figura 3.1: Diagrama de Blocos - TM4C1294NCPDT [1]

# Iniciando um projeto no Code Composer

Os projetos abordados adiante farão uso da IDE Code Composer que é baseada em Eclipse em sua versão 6.1.2 que é a mais recente no momento em que este texto é escrito. oferecida gratuitamente mediante a um cadastro realizado no site da Texas Instruments.

Na hora de instalar a IDE é preciso que sejam marcadas as opções de compatibilidade com a placa em uso, a Tiva C Series TM4C1294 Connected LaunchPad, e ainda seu compilador GCC, caso contrário o projeto não poderá ser criado.

Após iniciar o Code Composer, inicie um novo projeto em **File** > **New** > **CCS Project** como mostrado na Figura 4.1.



Figura 4.1: Criando um novo projeto

Uma janela de configurações será exibida para que o ambiente seja preparado para o hardware em uso, como na Figura 4.2.

Para o hardware aqui utilizado, em Target escolhe-se a opção Tiva C Series e no

segundo campo Tiva TM4C1294NCPDT.

Em Connection será utilizada a **Stellaris In-Circuit Debug Interface** para a programação e debug do microcontrolador.

Após isso, escolhe-se um nome para o projeto e o diretório que será armazenado, que é normalmente o local do workspace padrão marcando a opção **Use default location**.

A Texas Instruments disponibiliza um compilador próprio porém será usado aqui o GCC, compilador de código aberto sob a licença GNU. Portanto, em *Compile version* escolhe-se a opção **GNU** com a versão mais recente. As outras opções não precisam ser alteradas.



Figura 4.2: Configurando o projeto

Clicando em *Finish* o projeto será criado. Para o correto funcionamento do compilador GCC devem-se ainda ser feitos mais alguns ajustes.

Selecionando o projeto criado na barra lateral  $Project\ Explorer$ , vá em Project > Properties como na Figura 4.3.



Figura 4.3: Abrindo propriedades do projeto

Na janela de propriedades selecione **Build** > **GNU Compiler** > **Symbols**. Adicione um novo símbolo clicando no botão Add como na Figura 4.4. Na janela que se abre digite **TARGET\_IS\_TM4C129\_RA1** e clique em OK. Adicione ainda o símbolo **gcc**. Esses símbolos não podem conter erros de escrita, caso contrário causarão erros na hora da compilação. Ao se ter os três símbolos mostrados na Figura 4.4, selecione **Build** > **GNU Linker** > **Basic**. Na opção Set start address digite start start



Figura 4.4: Adicionando símbolo para a compilação no GCC



Figura  $4.5\colon$  Configurando endereço de início do Linker do GCC

Ao fim desses passos o projeto estará criado e poderá ser compilado no Code Composer utilizando o GCC.

### Biblioteca TivaWare

Para facilitar a programação do microcontrolador será feito o uso da biblioteca TivaWare fornecida pela Texas Instruments. Tal ferramenta facilita o controle do processador e acesso aos periféricos disponíveis. A TivaWare pode ser obtida no site da empresa gratuitamente.

O site disponibiliza somente a versão para o sistema Windows, que vem em formato executável, sendo preciso apenas dar um clique duplo sobre o arquivo e seguir os passos da instalação. Já para sistemas não derivados do *MS-DOS*, basta abrir este mesmo executável baixado com um aplicativo de descompactação de arquivos e copiar o conteúdo para um diretório qualquer desejado.

A estrutura do TivaWare é composta basicamente de dois diretórios:

driverlib/ Contém o código fonte para os drivers do dispositivo

inc/ Contém os arquivos de cabeçalho que são usados pelos drivers para acessar os registradores do microcontrolador

Os outros arquivos contidos no pacote do TivaWare são extras que facilitam alguns usos do microcontrolador. Como o diretório 'examples/' que contém códigos prontos para utilização em alguns dos microcontroladores e periféricos suportados, o 'utils/' com algumas implementações frequentes e a biblioteca 'usblib/' que implementa uma comunicação usb com portes para vários tipos de arquivos.

### 5.1 Incluindo a TivaWare ao projeto

Para a utilização da TivaWare nos projetos que serão apresentados é preciso que as aplicações desenvolvidas tenham acesso à tais bibliotecas. Tal comunicação pode ser feita de dois tipos: linkando ou copiando a biblioteca para o diretório do código fonte ou adicionando o diretório da biblioteca nos comandos de compilação.

#### 5.1.1 Bibliotecas junto ao código fonte

Este método pode ser feito de dois modos, copiando as bibliotecas para o diretório do código fonte da aplicação, ou *linkando*-as a este diretório.

É importante notar que se os arquivos de código fonte forem portados para outra máquina, somente serão compilados se as bibliotecas estiverem disponíveis nesta. Portanto, sempre que houver memória disponível, é aconselhável que se copie as bibliotecas usadas na aplicação para junto de seu diretório.

Para copiar as bibliotecas é possível apenas copiar as pastas para o diretório do projeto que este será atualizado automaticamente ou ainda arrastar e soltar o diretório ou arquivo da biblioteca sobre o projeto na barra lateral *Project Explorer* no Code Composer que será aberta uma janela intermediária como na figura 5.1.

Selecionando 'Copy files and folders' os arquivos serão copiados para o diretório do projeto escolhido. Já as duas outras opções criarão somente um link do arquivo no diretório especificado na caixa de seleção 'Create link location relative to', deste modo o compilador verá os arquivos como se eles estivessem neste diretório, porém existe apenas o caminho para alcançá-los. Se acaso eles forem movidos haverá erros de compilação.



Figura 5.1: Janela de importação de arquivos

#### 5.1.2 Inclusão de caminho na compilação

Um outro modo de juntar as bibliotecas ao código fonte é adicionando seu caminho à compilação. Com o projeto selecionado na janela lateral *Project Explorer*, vá em **Project** > **Properties** > **Build** > **GNU Compiler** > **Directories** e clique em *Add*, como na figura 5 2

Na janela aberta é possível digitar um caminho para o diretório ou arquivo, mas para prevenir erros existem os botões inferiores que abrirão uma navegação nos diretórios do sistema. Em **Workspace** é possível escolher o caminho para o diretório de um projeto ou de seus subdiretórios. Em **Variables** pode-se escolher o caminho armazenado em uma das variáveis de ambiente do projeto. E finalmente, em **Browse** é possível buscar um diretório navegando pelos arquivos do sistema.



Figura 5.3: Escolhendo diretórios para incluir na compilação



Figura 5.2: Incluindo diretórios para compilação

#### 5.2 TivaWare na ROM

O TM4C1294NCPDT possui carregado na memória ROM uma parte da biblioteca de drivers do TivaWare. Isso possibilita a geração de um arquivo menor na hora da compilação, economizando memória de programa.

Para o uso das funções gravadas na ROM é necessário importar o arquivo de cabeçalho 'driverlib/rom.h' e ainda usar o prefixo ' $ROM_{\_}$ ' junto a função desejada. Por exemplo, para usar a função de configuração de clock do sistema

carregada na ROM, esta deve ser chamada como

$$ROM$$
  $SysCtlClockFreqSet()$ .

Porém, ao chamar tal função da ROM é possível que ela não seja encontrada na hora da compilação. Isso se deve ao fato de que nem todos os hardwares compatíveis com a TivaWare possuem uma memória ROM carregada com sua biblioteca ou mesmo não possua toda ela. Tal problema é resolvido adicionando-se o arquivo de cabeçalho ' $\frac{driverlib}{rom_map.h}$ ' e usando o prefixo ' $\frac{MAP_{-}}{r}$ ' junto às funções ao invés de ' $\frac{ROM_{-}}{r}$ '. Para o exemplo da função de configuração de clock, a chamada seria feita da forma

$$MAP$$
  $SysCtlClockFreqSet().$ 

Esse arquivo de cabeçalho implementa uma estrutura que confere se a função usada existe na ROM do dispositivo para o qual o código será compilado e só assim a substitui. O prefixo de mapeamento pode ser usado em todas as chamadas de funções implementadas pela TivaWare.

### Sistema de Clock

Há múltiplas fontes de clock para o uso no microcontrolador. Elas devem ser configuradas logo após um *Power-On Reset* (POR), ou seja, quando o dispositivo é iniciado ou recuperado de um *reset*.

#### 6.1 Fontes de clock

As fontes disponíveis para o controle do TM4C1294NCPDT são:

#### Oscilador Interno de Precisão (do inglês, PIOSC)

Fonte de clock interna ao microcontrolador que é usada durante e logo após um POR. É o clock usado para iniciar a execução de uma aplicação. Não necessita de nenhum componente externo e fornece um clock de 16 MHz que apesar de ser preciso varia com temperaturas mais extremas. O PIOSC é útil também para aplicações que não exijam uma fonte de clock tão precisa. Mesmo sendo ou não o clock do sistema, o PIOSC pode ser configurado como fonte de clock de um periférico.

#### Oscilador Principal (do inglês, MOSC)

O oscilador principal fornece um clock de precisão por meio de um desses métodos: uma fonte *single-end* de clock é conectada ao pino de entrada OSC0 do microcontrolador, ou um cristal externo é conectado entre o pino de entrada OSC0 e o pino de saída OSC1. Com o PLL em uso, o cristal deve ser de uma frequência entre 5 MHz e 25 MHz. Se não, pode variar de 4 MHz até 25 MHz.

#### Oscilador Interno de Baixa-frequência (do inglês, LFIOSC)

Clock com frequência nominal de 33 KHz com uma porcentagem de variação. É usado durante os modos de economia de energia *Deep-Sleep*. Estes modos proveem um número reduzido de periféricos em funcionamento e podem desligar o MOSC e/ou PIOSC enquanto o microcontrolador está neste estado.

#### Oscilador RTC do Módulo de Hibernação (do inglês, RTCOSC)

Fornece uma saída de clock para o sistema selecionável entre duas: um clock externo de 32.768 Hz ou um Clock de Baixa Frequência de Hibernação (HIB LFIOSC). O Módulo de Hibernação pode receber um sinal de clock de 32.768 Hz conectado ao pino XOSCO. O oscilador de 32.768 Hz pode ser usado para o clock do sistema, assim eliminando a necessidade de um cristal ou oscilador adicional. Alternativamente, o Módulo de Hibernação contem um oscilador de baixa frequência (HIB LFIOSC) que provê um RTC para o sistema e pode também prover um clock de precisão para os modos de economia de energia Deep-Sleep ou Hibernação. Note que o HIB LFIOSC é uma fonte de clock diferente de LFIOSC, os dois possuem a mesma frequência nominal

mas, enquanto o primeiro pode variar de  $10~\mathrm{KHz}$  à  $90~\mathrm{KHz}$ , o segundo varia de  $10~\mathrm{KHz}$  à  $70~\mathrm{KHz}$ .

O clock interno do sistema (SysClk), pode ser derivado de qualquer uma das fontes anteriormente listadas. Um PLL interno pode ser usado pelo PIOSC ou pelo MOSC, e somente estes, para gerar o SysClk e os clocks dos periféricos.

#### 6.2 Circuito de verificação do MOSC

O microcontrolador possui circuitos de controle de clock que verificam se o Oscilador Principal está funcionando adequadamente e na frequência apropriada. O circuito sinaliza quando esta frequência se encontra fora dos valores permitidos para o cristal em uso.

Este circuito deve ser habilitado em tempo de execução. Quando ligado, se um erro for constatado, o MOSC é desligado, é ligado o PIOSC e o sistema é resetado levando o processador a uma interrupção não-mascarável (NMI).

#### 6.3 Na TivaWare

As principais funções de configuração do clock no TivaWare estão listadas abaixo.

```
void SysCtlMOSCConfigSet(uint32_t ui32Config)
```

Configura o circuito monitor do oscilador principal.

#### ui32Config

Configura o controle do oscilador principal a partir da lógica OR de definições no formato **SYSCTL MOSC** k, onde k pode assumir o valor de:

- VALIDATE para verificar uma falha do MOSC
- INTERRUPT quando se deseja gerar uma interrupção ao invés do reset do processador
- NO\_XTAL se não há um oscilador esxterno nos pinos OSC0/OSC1, reduzindo o consumo de energia
- PWR\_DIS se deseja-se que o MOSC seja desligado. Se este parâmetro não for especificado o oscilador permanece ligado
- LOWFREQ se a frequência do MOSC está abaixo de 10 MHZ
- HIGHFREQ se a frequência do MOSC está acima de 10 MHZ
- **SESRC** quando o MOSC é um oscilador *single-end* conectado ao pino OSC0. Se não especificado, assume-se que um cristal está em uso.

Configura o clock do sistema, frequência de entrada, seu oscilador fonte e o uso ou não do PLL. Retorna o valor da frequência definida em Hz.

#### ui32Config

Configuração do clock. Lógica OR das seguintes máscaras:

SYSCTL\_XTAL\_kMHZ indica o uso de um cristal externo de k MHz, podendo assumir os valores: 5, 6, 8, 10, 12, 16, 18, 20, 24 ou 25.

SYSCTL OSC k corresponde ao oscilador usado. Sendo k:

- MAIN para o oscilador principal
- INT para o oscilador interno de precisão de 16 MHz
- INT30 para o oscilador interno de baixa frequência
- EXT32 para o oscilador de 32.768 Hz do módulo de hibernação (apenas quando o módulo estiver disponível).

 ${\tt SYSCTL\_USE\_k}$  fonte do clock do sistema. Podendo k assumir os valores:

- PLL quando a saída do PLL fornece o clock do sistema.
- OSC para o oscilador fonte alimentar o clock do sistema.

SYSCTL\_CFG\_VCO\_k indica a frequência do VCO do PLL quando este está em uso. Os valores de k podem somente assumir os valores de 480 para 480 MHz e 320 para 320 MHz. O TivaWare escolhe o valor do divisor do PLL para gerar o clock mais próximo do valor desejado a partir destes valores de VCO.

#### ui32SysClock

Valor inteiro requerido para o clock do sistema. Se não for possível alcançá-lo com as configurações usadas é assumido o valor mais próximo de clock abaixo deste valor.

void SysCtlPeripheralEnable(uint32\_t ui32Peripheral)

Habilita o clock em um dos periféricos. Há um período de 5 ciclos de clock da chamada da função até a real ligação do periférico. Cuidados devem ser tomados para que não haja o acesso ao periférico durante este curto espaço de tempo.

#### ui32Peripheral

Periférico a ser ligado o clock.

SYSCTL PERIPH k indica o periférico à ser habilitado, onde k deve ser:

- ADCk para representar o AD de número k.
- $\mathbf{CAN}k$  para representar o barramento CAN de número k.
- CCMk para representar o módulo CRC do barramento CAN de número k.
- $\mathbf{COMP}k$  para representar o comparador de número k do AD.
- $\bullet$  **EEPROM** k para representar a EEPROM de número k
- EMAC para representar o módulo Ethernet MAC.
- EPHY para representar o módulo Ethernet PHY.
- EPI para representar a Interface de Periféricos Externos.
- **GPIO**k para representar a GPIO de letra k.
- HIBERNATE para representar o módulo de hibernação.
- $\mathbf{I2C}k$  para representar o módulo I2C de número k.
- PWMk para representar o módulo de PWM de número k.
- QEIk, QEI1 para representar de número k.
- $\mathbf{SSI}k$  para representar o módulo SSI de número k.
- $\mathbf{TIMER}k$  para representar o contador de número k.
- $\mathbf{UART}k$  para representar o módulo UART de número k.
- UDMA para representar o módulo de Acesso Direto à Memória.
- $\mathbf{USB}k$  para representar o módulo USB de número k.

- WDOGk para representar o módulo de estouro de tempo de número k.
- WTIMERk para representar o contador do módulo de estouro de tempo de número k.

#### void SysCtlPeripheralDisable(uint32\_t ui32Peripheral)

Desabilita o clock em um dos periféricos. Uma vez desabilitado, não responderá a nenhum comando.

#### ui32Peripheral

Periférico a ser desabilitado.

SYSCTL PERIPH k indica o periférico à ser habilitado, onde k deve ser:

- ADCk para representar o AD de número k.
- CANk para representar o barramento CAN de número k.
- ullet CCMk para representar o módulo CRC do barramento CAN de número k.
- COMPk para representar o comparador de número k do AD.
- **EEPROM**k para representar a EEPROM de número k.
- EMAC para representar o módulo Ethernet MAC.
- EPHY para representar o módulo Ethernet PHY.
- EPI para representar a Interface de Periféricos Externos.
- **GPIOk** para representar a GPIO de letra k.
- HIBERNATE para representar o módulo de hibernação.
- $\mathbf{I2C}k$  para representar o módulo I2C de número k.
- $\mathbf{PWM}k$  para representar o módulo de PWM de número k.
- $\mathbf{QEI}k$ ,  $\mathbf{QEI1}$  para representar de número k.
- $\mathbf{SSI}k$  para representar o módulo SSI de número k.
- $\mathbf{TIMER}k$  para representar o contador de número k.
- $\mathbf{UART}k$  para representar o módulo UART de número k.
- UDMA para representar o módulo de Acesso Direto à Memória.
- $\bullet$  USBk para representar o módulo USB de número k
- WDOGk para representar o módulo de estouro de tempo de número k.
- WTIMERk para representar o contador do módulo de estouro de tempo de número k.

#### void SysCtlPeripheralSleepEnable(uint32\_t ui32Peripheral)

Permite que o periférico continue operando mesmo quando o processador estiver em modo de economia de energia.

#### ui32Peripheral

Periférico a ser habilitado em modo de economia de energia.

SYSCTL PERIPH k indica o periférico à ser habilitado, onde k deve ser:

- ADCk para representar o AD de número k.
- $\mathbf{CAN}k$  para representar o barramento CAN de número k.
- ullet CCMk para representar o módulo CRC do barramento CAN de número k.

- $\mathbf{COMP}k$  para representar o comparador de número k do AD.
- **EEPROM**k para representar a EEPROM de número k.
- EMAC para representar o módulo Ethernet MAC.
- EPHY para representar o módulo Ethernet PHY.
- EPI para representar a Interface de Periféricos Externos.
- **GPIO**k para representar a GPIO de letra k.
- HIBERNATE para representar o módulo de hibernação.
- $\mathbf{I2C}k$  para representar o módulo I2C de número k.
- $\mathbf{PWM}k$  para representar o módulo de PWM de número k.
- $\mathbf{QEI}k$ ,  $\mathbf{QEI1}$  para representar de número k.
- $\mathbf{SSI}k$  para representar o módulo SSI de número k.
- $\mathbf{TIMER}k$  para representar o contador de número k.
- $\mathbf{UART}k$  para representar o módulo UART de número k.
- UDMA para representar o módulo de Acesso Direto à Memória.
- $\mathbf{USB}k$  para representar o módulo USB de número k.
- WDOGk para representar o módulo de estouro de tempo de número k.
- WTIMERk para representar o contador do módulo de estouro de tempo de número k.

#### void SysCtlPeripheralSleepDisable(uint32\_t ui32Peripheral)

Desabilita o periférico quando o processador estiver em modo de economia de energia. Isso ajuda a diminuir a corrente usada no dispositivo.

#### ui32Peripheral

Periférico a ser desabilitado em modo de economia de energia.

#### SYSCTL PERIPH k indica o periférico à ser habilitado, onde k deve ser:

- $\mathbf{ADC}k$  para representar o AD de número k.
- $\mathbf{CAN}k$  para representar o barramento CAN de número k.
- $\mathbf{CCM}k$  para representar o módulo CRC do barramento CAN de número k.
- COMPk para representar o comparador de número k do AD.
- **EEPROM**k para representar a EEPROM de número k.
- EMAC para representar o módulo Ethernet MAC.
- EPHY para representar o módulo Ethernet PHY.
- EPI para representar a Interface de Periféricos Externos.
- **GPIO**k para representar a GPIO de letra k.
- HIBERNATE para representar o módulo de hibernação.
- $\mathbf{I2C}k$  para representar o módulo I2C de número k.
- $\mathbf{PWM}k$  para representar o módulo de PWM de número k.
- $\mathbf{QEI}k$ ,  $\mathbf{QEI1}$  para representar de número k.
- SSIk para representar o módulo SSI de número k.
- $\mathbf{TIMER}k$  para representar o contador de número k.
- $\mathbf{UART}k$  para representar o módulo  $\mathbf{UART}$  de número k.
- UDMA para representar o módulo de Acesso Direto à Memória.
- $\mathbf{USB}k$  para representar o módulo USB de número k.
- WDOGk para representar o módulo de estouro de tempo de número k.

 WTIMERk para representar o contador do módulo de estouro de tempo de número k.

```
void SysCtlPeripheralClockGating(bool bEnable)
```

Habilita e desabilita o clock dos periféricos em modo de economia de energia.

#### bEnable

Valor booleano que deve ser **true** se os periféricos podem ser usados durante o modo de economia de energia, e **false** se eles não podem ser usados neste período.

#### 6.4 Exemplo

Um exemplo de configuração do clock do microcontrolador é dado a seguir:

```
1|\ //\  Configurando circuito de verificacao
 2 // do MOSC para frequencias acima de 10 MHZ
 3 SysCtlMOSCConfigSet(SYSCTL_MOSC_HIGHFREQ);
 5 // Fonte de clock externa de 25 MHz,
 6 // provinda do oscilador principal,
  // usando a saida do PLL com fvco = 480 MHz,
 8 // gerando um clock de 120 MHz para o microcontrolador
9 int systemClockFreq = SysCtlClockFreqSet ( \
10
               (SYSCTL_XTAL_25MHZ |
11
                SYSCTL_OSC_MAIN | \
12
                SYSCTL_USE_PLL | \
13
                SYSCTL_CFG_VCO_480), 120000000);
14
15 // Habilita o funcionamento da GPIO A e da GPIO B
16 SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOA);
17 SysCtlPeripheralEnable (SYSCTL_PERIPH_GPIOB);
18
19 // Habilita o uso somente da GPIO B em
20 // modo de economia de energia
21|\ //\ {\tt A} GPIO A nao podera ser utilizado durante
22 // este periodo
23 SysCtlPeripheralSleepEnable(SYSCTL_PERIPH_GPIOB);
24
25 // Habilita o clock nos perifericos em modo de economia de
      energia
26 SysCtlPeripheralClockGating(true);
```

# Portas de Entrada e Saída de Propósito Geral (GPIOs)

O TM4C1294NCPDT possui 15 portas GPIOs de 8 pinos cada. Elas são nomeadas com as letras de A à A menos as letras A e A letras de A a especificações das GPIOs são:

- Possui mais de 90 GPIOS, dependendo da configuração usada.
- Pinos específicos possuem ligação com os periféricos do microcontrolador e suas funções devem ser configuradas.
- Tensão em configuração de entrada de 3,3 V.
- Todas as portas são conectadas ao Barramento de Alta Performance (AHB).
- Mudança rápida de nível de saída da porta a cada ciclo de clock em portas ligadas ao
   AHR
- Interrupções por pinos nas portas P e Q por bordas de subida, descida ou ambas.
- $\bullet$  Podem ser usadas para iniciar uma sequência de amostragem do A/D ou uma transferência  $\mu {\rm DMA}.$
- Estado dos pinos podem ser mantidos durante o modo de hibernação; variações de nível nos pinos da porta P podem ser usadas para acordar o sistema da hibernação.
- Pinos configurados como entradas digital utilizam circuitos Schmitt-trigger.
- Pinos possuem resistores de pull-up e pull-down e limites de corrente para 2, 4, 6, 8, 10 e 12 mA.
- Configuração dreno-aberto habilitada

#### 7.1 Na TivaWare

As principais funções presentes na TivaWare responsáveis pela configuração e manipulação das GPIOs são listadas a seguir.

#### CAPÍTULO 7. PORTAS DE ENTRADA E SAÍDA DE PROPÓSITO GERAL (GPIOS)27

Configura a direção dos pinos da porta especificada.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO\_PORT}k\_\mathbf{BASE},$  onde k é a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato **GPIO PIN** k, onde k é o número do pino de 0 a 7.

#### ui32PinIO

Direção ou modo dos pinos especificados.

```
    GPIO_DIR_MODE_IN pinos configurados como de entrada
    GPIO_DIR_MODE_OUT pinos configurados como de saída
    GPIO_DIR_MODE_HW direção dos pinos controlada pelo pelo hardware de algum periférico
```

```
void GPIOPinConfigure(uint32_t ui32PinConfig)
```

Configura pino utilizado por algum periférico especial.

#### ui32PinConfig

Valor de configuração do pino, especificado pelas definições no formato  $\mathbf{GPIO}_{-}\mathbf{P}Ak_{-}pin$ . Sendo A uma letra representando a porta que contém o pino, k o número do pino referente à porta e pin a função atribuída ao pino.

```
uint32_t GPIODirModeGet(uint32_t ui32Port,
uint8_t ui8Pin)
```

Retorna a direção de pinos de uma determinada porta.

#### ui32Port

Base da porta a ser consultada. Normalmente  $\mathbf{GPIO}\_\mathbf{PORT}k\_\mathbf{BASE},$ onde ké a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato  $\mathbf{GPIO}_{\mathbf{PIN}_{\mathbf{k}}}$ , onde  $\mathbf{k}$  é o número do pino de 0 a 7.

Configura especificações de potência e tipo dos pinos das portas GPIOs.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO\_PORT}k\_\mathbf{BASE},$ onde ké a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato **GPIO PIN** k, onde k é o número do pino de 0 a 7.

#### ui32Strength

Especifica a corrente máxima que os pinos podem fornecer. Configurada através da definição **GPIO\_STRENGTH\_kMA** onde o máximo de corrente nos pinos é de k mA. Sendo que k pode assumir os valores: 2, 4, 6, 8, 10 e 12.

#### ui32PinType

Valor que configura o tipo do pino. Definido por  $\mathbf{GPIO}_{\mathbf{PIN}_{\mathbf{TYPE}_{\mathbf{k}}}}$ , onde k pode assumir os seguintes valores:

**STD** pino do tipo *push-pull* (tipo padrão)

 $\mathbf{STD}_{-}\mathbf{WPU}$ pino do tipo pull-up fraco, ou seja, com um resistor de pull-up pouca corrente circulando

STD\_WPD pino do tipo *pull-down* fraco, ou seja, com um resistor de *pull-down* pouca corrente circulando

**OD** pino do tipo coletor aberto

ANALOG pino de entrada analógica

Retorna as informações de configuração utilizadas no pino de uma determinada porta.

#### ui32Port

Base da porta a ser consultada. Normalmente  $\mathbf{GPIO}\_\mathbf{PORT}k\_\mathbf{BASE},$  onde k é a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato  $\mathbf{GPIO}_{\mathbf{PIN}_{\mathbf{k}}}$ , onde  $\mathbf{k}$  é o número do pino de 0 a 7.

#### pui32Strength

Ponteiro de um endereço de memória já alocado, onde estarão contidas as especificações de corrente.

#### pui32PinType

Ponteiro de um endereço de memória já alocado, onde estarão contidas as especificações do tipo do pino.

#### CAPÍTULO 7. PORTAS DE ENTRADA E SAÍDA DE PROPÓSITO GERAL (GPIOS)29

Retorna os valores contidos nos pinos de entrada especificados.

#### ui32Port

Base da porta a ser consultada. Normalmente  $\mathbf{GPIO}_{PORT}k_{BASE}$ , onde k é a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato **GPIO PIN** k, onde k é o número do pino de 0 a 7.

Configura os valores nos pinos de saída especificados.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO}_{\mathbf{PORT}k}_{\mathbf{BASE}}$ , onde k é a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato **GPIO PIN** k, onde k é o número do pino de 0 a 7.

#### ui32Val

Pacote de valores de cada pino, contendo 8 bits. Onde o bit 0 representa sinal alto no pino 0, o bit 1 representa sinal alto no pino 1 e assim por diante. É possível utilizar das definições no formato  $\mathbf{GPIO}_{\mathbf{PIN}_{\mathbf{k}}}$ , onde  $\mathbf{k}$  é o número do pino de 0 a 7.

Configura os tipos das interrupções nos pinos de entrada especificados.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO}_{\mathbf{PORT}k}_{\mathbf{BASE}}$ , onde k é a letra identificadora da porta.

#### ui32Pins

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato **GPIO PIN** k, onde k é o número do pino de 0 a 7.

#### ui32IntType

Definição do evento que dispara a interrupção.

```
    GPIO_FALLING_EDGE disparo de interrupção na borda de descida
    GPIO_RISING_EDGE disparo de interrupção na borda de subida
    GPIO BOTH EDGES disparo de interrupção em ambas as bordas
```

GPIO\_LOW\_LEVEL disparo de interrupção quando em nível baixoGPIO\_HIGH\_LEVEL disparo de interrupção quando em nível alto

Habilita as interrupções nos pinos especificados.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO}_{\mathbf{PORT}k}_{\mathbf{BASE}}$ , onde k é a letra identificadora da porta.

#### ui32IntFlags

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato  $\mathbf{GPIO\_INT\_PIN\_k}$ , onde  $\mathbf{k}$  é o número do pino de 0 a 7.

Desabilita as interrupções nos pinos especificados.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO\_PORT}k\_\mathbf{BASE},$ onde ké a letra identificadora da porta.

#### ${\tt ui32IntFlags}$

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato  $\mathbf{GPIO\_INT\_PIN\_k}$ , onde  $\mathbf{k}$  é o número do pino de 0 a 7.

Limpa o buffer que armazena se já houve uma interrupção nos pinos especificados.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO\_PORT}k\_\mathbf{BASE},$ onde ké a letra identificadora da porta.

#### ui32IntFlags

Pacote em OU binário de pinos com 8 bits. Onde o bit 0 representa o pino 0, o bit 1 representa o pino 1 e assim por diante. É possível utilizar das definições no formato **GPIO INT PIN** k, onde k é o número do pino de 0 a 7.

#### CAPÍTULO 7. PORTAS DE ENTRADA E SAÍDA DE PROPÓSITO GERAL (GPIOS)31

Configura a função de tratamento de uma interrupção da GPIO. Aquela que é chamada quando ocorre a interrupção.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO\_PORT}k\_\mathbf{BASE},$ onde ké a letra identificadora da porta.

#### pfnIntHandler

Ponteiro da função de tratamento. Esta não deve receber nada como parâmetro e nem retornar nada.

Configura a função de tratamento de uma interrupção da GPIO. Aquela que é chamada quando ocorre a interrupção.

#### ui32Port

Base da porta a ser configurada. Normalmente  $\mathbf{GPIO}_{\mathbf{PORT}k}_{\mathbf{BASE}}$ , onde k é a letra identificadora da porta.

#### pfnIntHandler

Ponteiro da função de tratamento. Esta não deve receber nada como parâmetro e nem retornar nada.

#### 7.2 Exemplo

Um exemplo básico de configuração das portas GPIOs do microcontrolador é dado a seguir:

```
1
2 // Configura os pinos 0, 2 e 7 da porta A como sendo de entrada
3 GPIODirModeSet(GPIO_PORTA_BASE, GPIO_PIN_0 | GPIO_PIN_2 |
     GPIO_PIN_7, GPIO_DIR_MODE_IN);
4
5
  // Configura os pinos 4, 5 e 6 da porta A como sendo de saida
6 GPIODirModeSet(GPIO_PORTA_BASE, GPIO_PIN_4 | GPIO_PIN_5 |
     GPIO_PIN_6, GPIO_DIR_MODE_OUT);
  // Configura os pinos 4 e 6 daporta A para poder fornecer at
     4 mA para o circuito
9 GPIOPadConfigSet(GPIO_PORTA_BASE, GPIO_PIN_4 | GPIO_PIN_6,
     GPIO_STRENGTH_4MA);
10
|11| // Armazena valor de entrada do pino 2 da porta A
12 int valor = GPIOPinRead(GPIO_PORTA_BASE, GPIO_PIN_2);
13
14 // Configura pino de sa da 4 com sinal alto e o 6 com com
     sinal alto
15 GPIOPinWrite(GPIO_PORTA_BASE, GPIO_PIN_4 | GPIO_PIN_6,
     GPIO_PIN_4 | ~(GPIO_PIN_6));
```

### **UART**

O Transmissor/Receptor Assíncrono Universal (*Universal Asynchronous Receiver/Transmitter*, UART), é um periférico de transmissão e recepção de dados usado na comunicação entre dispositivos, sendo esta comunicação realizada de forma serial e assíncrona, ou seja sem a necessidade de transmissão do sinal de clock de referência. Este modo de transmissão faz necessário o uso de apenas duas vias de comunicação uma para a transmissão e outra para a recepção de dados.

#### 8.1 Padrão da Comunicação



Figura 8.1: Protocolo de envio na comunicação UART

Para que a comunicação UART seja realizada é necessário que o sinal de transmissão obedeça a um protocolo. Quando uma palavra é transmitida, primeiro é enviado um bit de início de transmissão para o receptor. Este bit deve ser de nível logico 0 para que a ocorrência da borda de descida sinalize ao receptor que sincronize a amostragem do sinal a ser lido de modo que ela ocorra no meio de cada período de transmissão. Após transmitir os dados é necessário enviar um bit informando a existência de paridade ou não, e por último é enviado um bit de nível lógico alto para informar o fim da transmissão. Esta sintaxe pode ser observada na figura 8.1.

#### 8.2 UART do TM4C1294NCPDT

O Tiva TM4C1294NCPDT possui 8 módulos de comunicação UART. Cada um destes possuem um gerador de *baud-rate*, ou taxa de transmissão, que possibilitam transmissões de até 7,5 Mbps em modo de normal transmissão e 15 Mbps em modo *High Speed*.

33

Para que seja possível regular o *baud-rate* de forma mais precisa os módulos UART possuem um divisor de 22 bits, sendo 16 bits inteiros e 6 bits fracionários, pelo qual o módulo determina o período de transmissão de bit.

Já o buffer de leitura e transmissão do UART no Tiva tem um tamanho de 8 bits, porém para cada módulo existe uma FIFO de 16x8 bits tanto para transmissão quanto para recepção, sendo que o *trigger* de interrupção de estouro da FIFO é selecionável entre 1/8, 1/4, 1/2, 3/4, 7/8 ou 8/8.

O sinal de transmissão criado pelo UART do tiva pode transmitir dados seriais de 5,6,7 ou 8 bits de dados precedidos do bit de *Start* e acompanhados de um bit de paridade, se estiver habilitado, e 1 ou 2 bits de parada. A figura 8.2 apresenta o sinal característico da transmissão UART do Tiva.



Figura 8.2: Sinal de Transmissão UART no Tiva TM4C1294NCPDT

A figura 8.1 apresenta os pinos de entrada e saída do UART que podem ser usados no Tiva TM4C1294NCPDT.

| Tabela 8.1: Canais do UART - Tiva TM4C1294NCPDT [1] |             |      |        |                                     |  |  |
|-----------------------------------------------------|-------------|------|--------|-------------------------------------|--|--|
| Pino                                                | Mux/Função  | Tipo | Buffer | Descrição                           |  |  |
| UORx                                                | PA0 (1)     | I    | TTL    | UART Módulo 0, recepção do sinal    |  |  |
| U0Tx                                                | PA1 (1)     | О    | TTL    | UART Módulo 0, transmissão do sinal |  |  |
| U1Rx                                                | PB0 (1)     | I    | TTL    | UART Módulo 1, recepção do sinal    |  |  |
|                                                     | PQ4 (1)     |      |        |                                     |  |  |
| U1Tx                                                | PB1 (1)     | О    | TTL    | UART Módulo 1, transmissão do sinal |  |  |
| U2Rx                                                | PA6 (1)     | I    | TTL    | UART Módulo 2, recepção do sinal    |  |  |
|                                                     | PD4 (1)     |      |        |                                     |  |  |
| U2Tx                                                | PA7 (1)     | О    | TTL    | UART Módulo 2, transmissão do sinal |  |  |
|                                                     | PD5 (1)     |      |        |                                     |  |  |
| U3Rx                                                | PA4 (1) (1) | I    | TTL    | UART Módulo 3, recepção do sinal    |  |  |
|                                                     | PJ0 (1)     |      |        |                                     |  |  |
| U3Tx                                                | PA5 (1)     | О    | TTL    | UART Módulo 3, transmissão do sinal |  |  |
|                                                     | PJ1 (1)     |      |        |                                     |  |  |
| U4Rx                                                | PK0 (1)     | I    | TTL    | UART Módulo 4, recepção do sinal    |  |  |
|                                                     | PA2 (1)     |      |        |                                     |  |  |
| U4Tx                                                | PK1 (1)     | О    | TTL    | UART Módulo 4, transmissão do sinal |  |  |
|                                                     | PA3 (1)     |      |        |                                     |  |  |
| U5Rx                                                | PC6 (1)     | I    | TTL    | UART Módulo 5, recepção do sinal    |  |  |
| U5Tx                                                | PC7 (1)     | О    | TTL    | UART Módulo 5, transmissão do sinal |  |  |
| U6Rx                                                | PP0 (1)     | I    | TTL    | UART Módulo 6, recepção do sinal    |  |  |
| U6Tx                                                | PP1 (1)     | О    | TTL    | UART Módulo 6, transmissão do sinal |  |  |
| U7Rx                                                | PC4 (1)     | I    | TTL    | UART Módulo 7, recepção do sinal    |  |  |
| U7Tx                                                | PC5 (1)     | О    | TTL    | UART Módulo 7, transmissão do sinal |  |  |

Tabela 8.1: Canais do UART - Tiva TM4C1294NCPDT [1]

### 8.3 Na TivaWare

### SPI

A Interface de Periféricos Serial, ou SPI (Serial Peripheral Interface) nada mais é do que um dispositivo periférico usado na transmissão e recepção serial de dados. Sendo uma comunicação síncrona, a SPI necessita de uma fonte de clock de referência para se estabelecer, além de um sinal de chip select, ou CS, para ativar a recepção de dados no dispositivo receptor. Deste modo esta comunicação requer no minimo três vias de transmissão, sendo que a comunicação em cada barramento é unidirecional.

#### 9.1 Padrão da Comunicação

A comunicação SPI possui a maior taxa de transmissão, ou baud-rate, dentre os demais protocolos de comunicação usados em microcontroladores, podendo chegar a até a 66Mpbs em periféricos com o AT45BD0100D da Adesto. O que possibilita um baud-rate tão elevado é o fato de que nesta comunicação a recepção e a transmissão de dados é feita separadamente e de forma direta, sem a necessidade de se transmitir bits de inicio ou termino de transmissão, e ainda de modo que o controle da transmissão é realizado pelo sinal CS (Chip Select) e pelo sinal CLK (Clock). A figura 9.1 apresenta o padrão de uma comunicação SPI.



Figura 9.1: Padrão de Comunicação SPI

Para transmitir um dado de um dispositivo **Mestre** para um **Escravo** é necessário que o **Mestre** ative o sinal de CS do **Escravo** e fornecer a ele o sinal de clock de referência. Em seguida bit a bit deve ser transmitido pela porta MOSI *Master Output - Slave Input* de ambos os dispositivos.

Quando for necessário transmitir um dado de um **Escravo** para um **Mestre**, novamente o **Mestre** deve ativar o sinal de CS do **Escravo** e fornecer a ele o sinal de clock de referência, porém o dado será transmitido bit a bit pela porta MISO *Master Input - Slave Output* de ambos dispositivos.

CAPÍTULO 9. SPI 36



Figura 9.2: Diagrama de Comunicação SPI - Vários Escravos

A figura 9.2 apresenta um diagrama básico de uma comunicação entre **Mestre** e vários **Escravos** através dos de barramentos de dados e de clock em comum.

#### 9.2 SPI do TM4C1294NCPDT

No microcontrolador Tiva TM4C1294NCPDT há 4 módulos de comunicação SPI, os quais possuem um baud rate máximo de 2MHz e uma FIFO para transmissão e outra para recepção ambas com capacidade de 16x8 bits. É possível alternar a fonte de clock de referência da transmissão entre o clock padrão do sistema (SYSCLK) e o clock alternativo (ALTCLK), contando ainda com um divisor de clock de 8 bits, que possibilita dividir o clock de 1 até 254 vezes.

Como é comum na comunicação SPI o Tiva possui portas para transmissão, recepção e clock exclusiva para cada módulo de comunicação. A tabela 9.1 apresenta as referidas portas para comunicação SPI.

| Tabela 9.1. Canais do SF1 - 11va 1 w 4C1294 NCFD1 [1] |            |      |        |                              |  |  |  |  |
|-------------------------------------------------------|------------|------|--------|------------------------------|--|--|--|--|
| Pino                                                  | Mux/Função | Tipo | Buffer | Descrição                    |  |  |  |  |
| SSI0CLK                                               | PA2 (15)   | I/O  | TTL    | SPI Módulo 0, sinal de clock |  |  |  |  |
| SSI0XDAT0                                             | PA4 (15)   | I/O  | TTL    | SPI Módulo 0, MISO           |  |  |  |  |
| SSI0XDAT1                                             | PA5 (15)   | I/O  | TTL    | SPI Módulo 0, MOSI           |  |  |  |  |
| SSI1CLK                                               | PB5 (15)   | I/O  | TTL    | SPI Módulo 1, sinal de clock |  |  |  |  |
| SSI1XDAT0                                             | PE4 (15)   | I/O  | TTL    | SPI Módulo 1, MISO           |  |  |  |  |
| SSI1XDAT1                                             | PE5 (15)   | I/O  | TTL    | SPI Módulo 1, MOSI           |  |  |  |  |
| SSI2CLK                                               | PB5 (15)   | I/O  | TTL    | SPI Módulo 2, sinal de clock |  |  |  |  |
| SSI2XDAT0                                             | PD1 (15)   | I/O  | TTL    | SPI Módulo 2, MISO           |  |  |  |  |
| SSI2XDAT1                                             | PD0 (15)   | I/O  | TTL    | SPI Módulo 2, MOSI           |  |  |  |  |
| SSI3CLK                                               | PQ0 (14)   | I/O  | TTL    | SPI Módulo 3, sinal de clock |  |  |  |  |
|                                                       | PF3 (14)   |      |        |                              |  |  |  |  |
| SSI3XDAT0                                             | PQ2 (14)   | I/O  | TTL    | SPI Módulo 3, MISO           |  |  |  |  |
|                                                       | PF1 (14)   |      |        |                              |  |  |  |  |
| SSI3XDAT1                                             | PQ3 (14)   | I/O  | TTL    | SPI Módulo 3, MOSI           |  |  |  |  |
|                                                       | PF0 (14)   |      |        |                              |  |  |  |  |

Tabela 9.1: Canais do SPI - Tiva TM4C1294NCPDT [1]

CAPÍTULO 9. SPI 37

### 9.3 Na TivaWare

### ADC

O ADC (Analog-To-Digital Converter) é um periférico responsável por realizar a conversão de uma grandeza analógica de tensão para um valor correspondente digital. Para realizar esta conversão pode-se implementar vários tipos circuito, como o conversor flash, ou o conversor de aproximações sucessivas, ou ainda conversor integrador simples ou de rampa dupla. Porém o circuito de conversão mais usado em circuito integrados atualmente é o conversor de aproximações sucessivas, o qual também usado como AD no Tiva TM4C1294NCPDT.

#### 10.1 ADC de Aproximações Sucessivas



Figura 10.1: Conversor A/D tipo Aproximação Sucessiva

A figura 10.1 apresenta o diagrama básico de funcionamento de um conversor de aproximações sucessivas. Nota-se que tal conversor utiliza a técnica de realimentação para relacionar uma voltagem analógica de entrada com um código digital, através de um conversor DA (Digital-To-Analog Converter) e um comparador. O processo de conversão inicia quando o Shift Register e o Holding Register são zerados, e então o MSB (Most Significant Bit) do Holding Register vai para nível alto. Em seguida o comparador relaciona a saída do conversor DA com a tensão  $V_{IN}$ . Se  $V_O < V_{IN}$  a conversão chega ao fim, porém se isso não for verdade a etapa se repete e MSB vai para nível baixo e o segundo SB vai para nível alto. E assim se dá a conversão.

#### 10.2 ADC do TM4C1294NCPDT

O Tiva TM4C1294NCPDT possui 2 módulos de conversão AD de 12-bit, que podem ser usados em qualquer um das 20 entradas de sinal analógico. É possível realizar amostragens sequenciais entre os canais ou no mesmo canal repetidamente com intervalos de tempo programáveis. Cada módulo AD possui 8 comparadores que possibilitam realizar comparações entre os sinais de entrada e valores pré-definidos, para que assim possa ser realizado as mais diversas operações. Ainda é possível usar um trigger diferente para cada um dos módulos, ou usar um trigger para acionar ambos os módulos.

A tabela 10.1 apresenta os pinos de entrada para os módulos ADC0 e ADC1, com a descrição do nome do pino de entrada, o numero referente a este pino, sua função e o tipo de buffer usado. Nesta mesma tabela temos o pino chamado de VREFA+ que é o pino da tensão de referência usado pelo AD. A VREFA+ corresponde o valor máximo que o conversor DA, usado pelo AD para realizar a comparação com a tensão de entrada, pode atingir.

A tensão VREFA+ é extremamente importante para se realizar a conversão AD, pois se este valor não for selecionado de forma adequada pode-se acarretar problemas no valor digital. A tensão VREFA+ pode ser alternada entre uma fonte de referência interna ou externa.

| Pino         | $n^o$    | Mux/Função | Tipo      | Buffer                  | Descrição                       |
|--------------|----------|------------|-----------|-------------------------|---------------------------------|
| AIN0         | 12       | PE3        | I         | Analógico               | ADC - Entrada 0                 |
| AIN1         | 13       | PE2        | I         | Analógico               | ADC - Entrada 0                 |
| AIN1<br>AIN2 | 14       | PE1        | I         |                         | ADC - Entrada 1 ADC - Entrada 2 |
|              |          |            | _         | Analógico               |                                 |
| AIN3         | 15       | PE0        | I         | Analógico               | ADC - Entrada 3                 |
| AIN4         | 128      | PD7        | I         | Analógico               | ADC - Entrada 4                 |
| AIN5         | 127      | PD6        | I         | Analógico               | ADC - Entrada 5                 |
| AIN6         | 126      | PD5        | I         | Analógico               | ADC - Entrada 6                 |
| AIN7         | 125      | PD4        | I         | Analógico               | ADC - Entrada 7                 |
| AIN8         | 124      | PE5        | I         | Analógico               | ADC - Entrada 8                 |
| AIN9         | 123      | PE4        | I         | Analógico               | ADC - Entrada 9                 |
| AIN10        | 121      | PB4        | I         | Analógico               | ADC - Entrada 10                |
| AIN11        | 120      | PB5        | I         | Analógico               | ADC - Entrada 11                |
| AIN12        | 4        | PD3        | I         | Analógico               | ADC - Entrada 12                |
| AIN13        | 3        | PD2        | I         | Analógico               | ADC - Entrada 13                |
| AIN14        | 2        | PD1        | I         | Analógico               | ADC - Entrada 14                |
| AIN15        | 1        | PD0        | I         | Analógico               | ADC - Entrada 15                |
| AIN16        | 18       | PK0        | I         | Analógico               | ADC - Entrada 16                |
| AIN17        | 19       | PK1        | I         | Analógico               | ADC - Entrada 17                |
| AIN18        | 20       | PK2        | I         | Analógico               | ADC - Entrada 18                |
| AIN19        | 21       | PK3        | I         | Analógico               | ADC - Entrada 19                |
|              |          |            |           |                         | A tensão de referência          |
| VREFA+       | + 9 fixo | fivo       |           | A no lá mi              | é usada pelo AD para            |
| V REFA+      |          | -          | Analógico | fixar o valor máximo de |                                 |
|              |          |            |           |                         | conversão.                      |
|              |          |            |           | D 0                     |                                 |

Tabela 10.1: Canais de Entrada ADC - Tiva TM4C1294NCPDT

### PWM

PWM (Pulse Width Modulation) é uma modulação baseada na conversão linear de um valor em escala de tensão para outro em escala de *Duty Cicle* aplicado a uma onda quadrada de amplitude qualquer. Este tipo de modulação é essencial em controladores de circuitos de potência, já que através desta modulação é possível controlar um dispositivo de chaveamento como um Mosfet, de modo que a tensão média sobre uma dada carga esteja diretamente relacionada com *Duty Cicle* do chaveamento.

#### 11.1 Modos de Funcionamento

Para criar uma modulação PWM é necessário criar uma onda portadora cuja a amplitude do sinal varie no tempo permitindo compara a sua amplitude com o do sinal modulante. O sinal modulante nada mais é do que o sinal ao qual se deseja converter em escala de *Duty Cicle*. Normalmente a onda portadora utilizada neste tipo de modulação é a onda triangular ou a onda dente de serra.



Figura 11.1: PWM modo Down [1]



Figura 11.2: PWM modo Down [1]

#### 11.2 PWM do TM4C1294NCPDT



Figura 11.3: PWM modo Down [1]



Figura 11.4: PWM modo Down [1]

# Exemplos de aplicação

### 12.1 Echo da UART

Tabela 12.1: Características Básicas - TM4C1294NCPDT [1]

| Pin Name | Pin Number |         | Pin Type | Buffer Type |             |
|----------|------------|---------|----------|-------------|-------------|
| U0RX     | 33         | PA0 (1) | I        | TTL         | UART module |
|          |            | (-)     |          |             | 0 receive.  |
| U0TX     | 34         | PA1 (1) | O        | TTL         | UART module |
|          |            |         |          |             | 0 transmit. |
| U1RX     | 95         | PB0 (1) | I        | TTL         | UART module |
|          | 102        | PQ4 (1) |          |             | 1 receive.  |
| U1TX     | 96         | PB1 (1) | О        | TTL         | UART module |
|          |            |         |          |             | 1 transmit. |
| U2RX     | 40         | PA6 (1) | I        | TTL         | UART module |
|          | 125        | PD4 (1) |          |             | 2 receive.  |
| U2TX     | 41         | PA7 (1) | O        | TTL         | UART module |
|          | 126        | PD4 (1) |          |             | 2 transmit. |
| U3RX     | 37         | PA4 (1) | I        | TTL         | UART module |
|          | 116        | PJ0 (1) |          |             | 3 receive.  |
| U3TX     | 38         | PA5 (1) | O        | TTL         | UART module |
|          | 117        | PJ1 (1) |          |             | 3 transmit. |
| U4RX     | 18         | PK0 (1) | I        | TTL         | UART module |
|          | 35         | PA2 (1) |          |             | 4 receive.  |
| U4TX     | 19         | PA5 (1) | О        | TTL         | UART module |
|          | 36         | PJ1 (1) |          |             | 3 transmit. |
| U5RX     | 23         | PC6 (1) | I        | TTL         | UART module |
|          |            |         |          |             | 5 receive.  |
| U5TX     | 22         | PC7 (1) | O        | TTL         | UART module |
|          |            |         |          |             | 5 transmit. |
| U6RX     | 118        | PP0 (1) | I        | TTL         | UART module |
|          |            |         |          |             | 6 receive.  |
| U6TX     | 119        | PP1 (1) | О        | TTL         | UART module |
|          |            |         |          |             | 6 transmit. |
| U7RX     | 25         | PC4 (1) | I        | TTL         | UART module |
|          |            |         |          |             | 7 receive.  |
| U7TX     | 24         | PC5 (1) | О        | TTL         | UART module |
|          |            |         |          |             |             |

```
#include <stdint.h>
2 #include <stdbool.h>
  #include "inc/hw_ints.h"
4 #include "inc/hw_memmap.h"
  #include "driverlib/rom.h"
6 #include "driverlib/rom_map.h"
  #include "driverlib/sysctl.h"
8 #include "driverlib/uart.h"
  #include "driverlib/pin_map.h"
10 #include "driverlib/gpio.h"
12 // Altere para o tamanho do buffer de entrada desejado
  #define INP_BUF_SIZ 20
  void UARTConfigure() {
16
    // Habilita GPIO A usado na comunicacao da UART O
18
    MAP_SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOA);
    // Aguarda 3 SysCtlDelay. Aproximadamente 10 ciclos de clock
20
    MAP_SysCtlDelay(3);
    // Configura PAO no modo Rx da UART O
    MAP_GPIOPinConfigure(GPIO_PAO_UORX);
    // Configura PA1 no modo Tx da UART O
24
    MAP_GPIOPinConfigure(GPIO_PA1_UOTX);
26
    // Habilita UART 0
28
    MAP_SysCtlPeripheralEnable(SYSCTL_PERIPH_UARTO);
    // Configura PAO e PA1 como pinos de comunicacao da UART
30
    MAP_GPIOPinTypeUART(GPIO_PORTA_BASE, GPIO_PIN_0 | GPIO_PIN_1);
    // Configura UART 0 com fonte de clock 120MHz para 115.200 baud 8N1
32
    MAP_UARTConfigSetExpClk(UARTO_BASE, 120000000, 115200, (
      UART_CONFIG_WLEN_8 | UART_CONFIG_STOP_ONE | UART_CONFIG_PAR_NONE))
34
    // Habilita interrupcoes no processador
36
    MAP_IntMasterEnable();
    // Habilita interrupcao da UART 0
38
    MAP_IntEnable(INT_UARTO);
    // Configura pinos de interrupcao da UART 0
    MAP_UARTIntEnable(UARTO_BASE, UART_INT_RX | UART_INT_RT);
40
42
  void UARTIntHandler(void) {
44
    uint32_t statusInterrupt;
    uint8_t buffer;
46
    // Salva o status de interrupcao da UART 0
    statusInterrupt = MAP_UARTIntStatus(UARTO_BASE, true);
48
    // Limpa interrupcoes encontradas na UART 0
    MAP_UARTIntClear(UARTO_BASE, statusInterrupt);
54
    // Enquanto houver caracteres na FIFO de transmissao para serem
      enviados
    while (MAP_UARTCharsAvail(UARTO_BASE)) {
```

```
56
      // Le proximo caractere recebido
      buffer = MAP_UARTCharGetNonBlocking(UARTO_BASE);
58
      // Envia caractere lido de volta para a UART 0
      MAP_UARTCharPutNonBlocking(UARTO_BASE, buffer);
60
    }
62 }
64 int main(void) {
    MAP_SysCtlMOSCConfigSet(SYSCTL_MOSC_HIGHFREQ);
    MAP_SysCtlClockFreqSet((SYSCTL_XTAL_25MHZ | SYSCTL_OSC_MAIN |
68
      SYSCTL_USE_PLL | SYSCTL_CFG_VCO_480), 120000000);
70
     // Funcao de inicializacao da UART
72
    UARTConfigure();
    while (1);
74
    return 0;
  }
```

Código 12.1: Código de exemplo

# Referências

[1] T. I. Incorporated.  $Tiva^{7M}$  TM4C1294NCPDT Microcontroller - DATA SHEET.