Comunicação SPI em Linux

spi em linux
Este post faz parte da série Device Drivers para Linux Embarcado

O protocolo SPI

Dando continuidade aos artigos sobre acesso a dispositivos típicos de sistemas embarcados no Linux, vamos abordar neste artigo Comunicação SPI em Linux. Por enquanto apenas visando softwares em espaço de usuário.

O protocolo SPI (Serial Peripheral Interface) foi definido pela Motorola. É um protocolo serial síncrono semelhante ao protocolo I2C, mas que utiliza três fios para comunicação e pode alcançar velocidades superiores ao I2C. A interface física é composta pelos sinais MOSI, MISO e SCLK que ligam a todos os dispositivos na forma de barramento. Pode existir também um quarto fio para seleção do dispositivo com o qual a comunicação será feita.

Assim como o I2C, o SPI apresenta o conceito de mestre e escravo da comunicação. O mestre inicia a transferência de dados e controla o sinal de clock para estabelecer o sincronismo. A transferência de dados processa-se em full-duplex sobre os sinais MOSI (Master Output Slave Input) e MISO (Master Input Slave Output). A taxa de transferência é definida pelo processador, no papel de mestre, através do sinal SCLK. A seleção do periférico é feita através do sinal SSn. Nessa configuração, pode haver um mestre e vários escravos no barramento SPI, como mostrado na Figura 1.

Comunicação SPI em Linux
Figura 1 – Ligação entre um mestre e escravos no barramento SPI.

O fluxo de dados é controlado pelo mestre através do sinal de clock SCLK. Só há transferência enquanto o mestre pulsar o sinal SCLK. Em repouso o sinal SCLK encontra-se estável com o valor lógico definido por CPOL.

Habilitando o driver SPI em Linux

Assim como para o I2C, o Linux disponibiliza um driver genérico para SPI. Ele cria entradas no diretório /dev para que o usuário possa acessar o dispositivo SPI como um arquivo através da funções open(), close(), read() e write(). Por ser um driver genérico, ele pode não atender a todos os requisitos do seu projeto. Se você pretende usar o chipset ENC28J60 para construir uma interface ethernet, por exemplo, você precisará de um driver específico. Para esse chipset já existe um driver no Linux. Mas, para um outro chipset que não tenha suporte no Linux, você deverá desenvolvê-lo.

Saiba que, para se comunicar com um dispositivo SPI em Linux, por meio da sua infraestrutura, o processador da placa que você está usando precisa ter o driver do controlador SPI presente no kernel. No artigo Linux Device Drivers – Diferenças entre Drivers de Plataforma e de Dispositivo explico em detalhes o funcionamento geral dos drivers. Consulte essa referência!

Agora vamos habilitar o driver SPI no kernel. Selecione Device Driver:

Habilitando o driver de SPI (menu Device Drivers).
Figura 2 – Habilitando o driver de SPI (menu Device Drivers).

Habilite o suporte a SPI apertando ‘y’, depois aperte ENTER:

Habilitando o driver de SPI (item SPI support)
Figura 3 – Habilitando o driver de SPI (item SPI support).

Habilite o suporte ao driver SPI em Linux como módulo apertando ‘m’ ou ‘y’ para deixar o driver embutido no kernel:

Habilitando o driver de SPI (módulo ou built-in).
Figura 4 – Habilitando o driver de SPI (módulo ou built-in).

Agora basta compilar o kernel!

Comunicação SPI em Linux

Agora veremos um exemplo de software para escrever e ler um dispositivo qualquer que utiliza o protocolo SPI. O dispositivo mais simples para fazer um teste é uma memória Flash. Mas pode ser usado qualquer outro dispositivo, como um módulo RFID.

Exemplo de software para SPI:

/*
 * Exemplo de software para comunicacao SPI
 * 
 * Autor: Vinicius Maciel
 *
 * comando para compilacao: gcc spi_teste.c -o spi_teste
 * 
 * comando para cross compilacao: arm-linux-gnueabihf-gcc spi_teste.c -o spi_teste
 */
 
#include <stdint.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <linux/types.h>
 
#include "spidev.h"
 
#define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0]))
 
/* Funcao que imprime mensagem de erro e aborta o programa
 * 
 */
static void pabort(const char *s)
{
    perror(s);
    abort();
}
 
// Nome do dispositivo de driver SPI no diretorio /dev 
static const char *device = "/dev/spidev0.0"; // (Mude de acordo com sua placa)
// Modo de operacao do controlador SPI do SoC
static uint32_t mode;
// Quantidade de bits da palavra de transferencia SPI
static uint8_t bits = 8;
// Velocidade de comunicao
static uint32_t speed = 1000000;
// Delay entre bytes transferidos, caso necessario
static uint16_t delay;
 
/*
 * Funcao que realiza uma transferencia full duplex,
 * ou seja, que escreve e ler ao mesmo tempo
 * 
 * Parametros ----
 * fd: inteiro retornado pela funcao open()
 * tx: para de escrita para mensagem SPI
 * rx: buffer de leitura de mensagem SPI
 */
static void transfer(int fd, uint8_t *tx, uint8_t *rx)
{
    int ret;
    
    // Estrutura que contem as informacoes para a transmissao da mensagem
    struct spi_ioc_transfer tr = {
	    .tx_buf = (unsigned long)tx,
	    .rx_buf = (unsigned long)rx,
	    .len = ARRAY_SIZE(tx),
	    .delay_usecs = delay,
	    .speed_hz = speed,
	    .bits_per_word = bits,
    };	
 
    // Funcao que faz a transferencia full duplex
    ret = ioctl(fd, SPI_IOC_MESSAGE(1), &tr);
    if (ret < 1)
	    pabort("Nao foi possivel enviar mensagem SPI");
 
    // Imprimi a mensagem recebida, caso haja alguma
    for (ret = 0; ret < ARRAY_SIZE(rx); ret++) {	    
	    printf("%x ", rx[ret]);
    }
    puts("");
}
 
 
int main(int argc, char *argv[])
{
    int ret = 0, i;
    int fd;
    // Buffers para leitura e escrita da mensagem SPI
    uint8_t tx_buffer[32], rx_buffer[32];
    
 
    // Abri o dispositivo de driver SPI e retorna um inteiro
    fd = open(device, O_RDWR);
    if (fd < 0)
	pabort("Erro ao abrir o dispositivo");
 
    // Escolha outros modos de operacao que o SPI da sua placa suporta
    mode = SPI_CPHA | SPI_CPOL | SPI_MODE_0;
    
    // Configura o modo de operacao
    ret = ioctl(fd, SPI_IOC_WR_MODE32, &mode);
    if (ret == -1)
	    pabort("Erro ao setar o modo do SPI");
 
    ret = ioctl(fd, SPI_IOC_RD_MODE32, &mode);
    if (ret == -1)
	    pabort("Erro ao setar o modo do SPI");
 
    // Configura o tamanho da palavra de transferencia SPI para escrita
    ret = ioctl(fd, SPI_IOC_WR_BITS_PER_WORD, &bits);
    if (ret == -1)
	    pabort("Erro ao setar os bits por palavra");
 
    // Configura o tamanho da palavra de transferencia SPI para leitura
    ret = ioctl(fd, SPI_IOC_RD_BITS_PER_WORD, &bits);
    if (ret == -1)
	    pabort("Erro ao ler os bits por palavra");
 
    // Configura a maxima velocidade de transferencia para escrita
    ret = ioctl(fd, SPI_IOC_WR_MAX_SPEED_HZ, &speed);
    if (ret == -1)
	    pabort("Erro ao setar a velocidade maxima em HZ");
 
    // Configura a maxima velocidade de transferencia para leitura
    ret = ioctl(fd, SPI_IOC_RD_MAX_SPEED_HZ, &speed);
    if (ret == -1)
	    pabort("Erro ao ler a velocidade maxima em HZ");
 
    // Imprimi informacoes de configuracao do SPI
    printf("Modo SPI: 0x%x\n", mode);
    printf("bits por palavra: %d\n", bits);
    printf("Maxima velocidade: %d Hz (%d KHz)\n", speed, speed/1000);
    
    // Preenche o buffer de transferencia com numeros de 0 a 9
    for (i = 0; i < 9; i++)
	tx_buffer[i] = i;
 
    // Faz a transferencia full duplex
    transfer(fd, tx_buffer, rx_buffer);
 
    // Fecha o dispositivo de driver SPI
    close(fd);
 
    return ret;
}

Para que o código seja compilado, é necessário incluir o seguinte código de cabeçalho:

/*
 * include/linux/spi/spidev.h
 *
 * Copyright (C) 2006 SWAPP
 *	Andrea Paterniani <a.paterniani@swapp-eng.it>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  */
 
#ifndef SPIDEV_H
#define SPIDEV_H
 
#include <linux/types.h>
 
/* User space versions of kernel symbols for SPI clocking modes,
 * matching <linux/spi/spi.h>
 */
 
#define SPI_CPHA		0x01
#define SPI_CPOL		0x02
 
#define SPI_MODE_0		(0|0)
#define SPI_MODE_1		(0|SPI_CPHA)
#define SPI_MODE_2		(SPI_CPOL|0)
#define SPI_MODE_3		(SPI_CPOL|SPI_CPHA)
 
#define SPI_CS_HIGH		0x04
#define SPI_LSB_FIRST		0x08
#define SPI_3WIRE		0x10
#define SPI_LOOP		0x20
#define SPI_NO_CS		0x40
#define SPI_READY		0x80
#define SPI_TX_DUAL		0x100
#define SPI_TX_QUAD		0x200
#define SPI_RX_DUAL		0x400
#define SPI_RX_QUAD		0x800
 
/*---------------------------------------------------------------------------*/
 
/* IOCTL commands */
 
#define SPI_IOC_MAGIC			'k'
 
/**
 * struct spi_ioc_transfer - describes a single SPI transfer
 * @tx_buf: Holds pointer to userspace buffer with transmit data, or null.
 *	If no data is provided, zeroes are shifted out.
 * @rx_buf: Holds pointer to userspace buffer for receive data, or null.
 * @len: Length of tx and rx buffers, in bytes.
 * @speed_hz: Temporary override of the device's bitrate.
 * @bits_per_word: Temporary override of the device's wordsize.
 * @delay_usecs: If nonzero, how long to delay after the last bit transfer
 *	before optionally deselecting the device before the next transfer.
 * @cs_change: True to deselect device before starting the next transfer.
 *
 * This structure is mapped directly to the kernel spi_transfer structure;
 * the fields have the same meanings, except of course that the pointers
 * are in a different address space (and may be of different sizes in some
 * cases, such as 32-bit i386 userspace over a 64-bit x86_64 kernel).
 * Zero-initialize the structure, including currently unused fields, to
 * accommodate potential future updates.
 *
 * SPI_IOC_MESSAGE gives userspace the equivalent of kernel spi_sync().
 * Pass it an array of related transfers, they'll execute together.
 * Each transfer may be half duplex (either direction) or full duplex.
 *
 *	struct spi_ioc_transfer mesg[4];
 *	...
 *	status = ioctl(fd, SPI_IOC_MESSAGE(4), mesg);
 *
 * So for example one transfer might send a nine bit command (right aligned
 * in a 16-bit word), the next could read a block of 8-bit data before
 * terminating that command by temporarily deselecting the chip; the next
 * could send a different nine bit command (re-selecting the chip), and the
 * last transfer might write some register values.
 */
struct spi_ioc_transfer {
	__u64		tx_buf;
	__u64		rx_buf;
 
	__u32		len;
	__u32		speed_hz;
 
	__u16		delay_usecs;
	__u8		bits_per_word;
	__u8		cs_change;
	__u8		tx_nbits;
	__u8		rx_nbits;
	__u16		pad;
 
	/* If the contents of 'struct spi_ioc_transfer' ever change
	 * incompatibly, then the ioctl number (currently 0) must change;
	 * ioctls with constant size fields get a bit more in the way of
	 * error checking than ones (like this) where that field varies.
	 *
	 * NOTE: struct layout is the same in 64bit and 32bit userspace.
	 */
};
 
/* not all platforms use <asm-generic/ioctl.h> or _IOC_TYPECHECK() ... */
#define SPI_MSGSIZE(N) \
	((((N)*(sizeof (struct spi_ioc_transfer))) < (1 << _IOC_SIZEBITS)) \
		? ((N)*(sizeof (struct spi_ioc_transfer))) : 0)
#define SPI_IOC_MESSAGE(N) _IOW(SPI_IOC_MAGIC, 0, char[SPI_MSGSIZE(N)])
 
 
/* Read / Write of SPI mode (SPI_MODE_0..SPI_MODE_3) (limited to 8 bits) */
#define SPI_IOC_RD_MODE			_IOR(SPI_IOC_MAGIC, 1, __u8)
#define SPI_IOC_WR_MODE			_IOW(SPI_IOC_MAGIC, 1, __u8)
 
/* Read / Write SPI bit justification */
#define SPI_IOC_RD_LSB_FIRST		_IOR(SPI_IOC_MAGIC, 2, __u8)
#define SPI_IOC_WR_LSB_FIRST		_IOW(SPI_IOC_MAGIC, 2, __u8)
 
/* Read / Write SPI device word length (1..N) */
#define SPI_IOC_RD_BITS_PER_WORD	_IOR(SPI_IOC_MAGIC, 3, __u8)
#define SPI_IOC_WR_BITS_PER_WORD	_IOW(SPI_IOC_MAGIC, 3, __u8)
 
/* Read / Write SPI device default max speed hz */
#define SPI_IOC_RD_MAX_SPEED_HZ		_IOR(SPI_IOC_MAGIC, 4, __u32)
#define SPI_IOC_WR_MAX_SPEED_HZ		_IOW(SPI_IOC_MAGIC, 4, __u32)
 
/* Read / Write of the SPI mode field */
#define SPI_IOC_RD_MODE32		_IOR(SPI_IOC_MAGIC, 5, __u32)
#define SPI_IOC_WR_MODE32		_IOW(SPI_IOC_MAGIC, 5, __u32)
 
#endif /* SPIDEV_H */

Este artigo foi originalmente publicado no site Software Livre.

Saiba mais sobre SPI

– Comunicação SPI 

– Linux Kernel Documentation

– Gerando PWM com a Raspberry PI

Device Drivers para Linux Embarcado

Exemplo de driver para Linux Embarcado Exemplo de Device Driver I2C para Linux Embarcado
Licença Creative Commons Esta obra está licenciada com uma Licença Creative Commons Atribuição-CompartilhaIgual 4.0 Internacional.
Comentários:
Notificações
Notificar
6 Comentários
recentes
antigos mais votados
Inline Feedbacks
View all comments
Mário Nunes
Mário Nunes
09/06/2015 17:43

Parabéns, muito bom seu artigo.

vinifr
vinifr
Reply to  Mário Nunes
10/06/2015 13:06

Valeu Mário! Espero escrever outros depois. 😀

Ronaldo Lins
Ronaldo Lins
07/04/2015 08:17

Excelente! Parabéns pela iniciativa Vinicius.

vinifr
vinifr
Reply to  Ronaldo Lins
07/04/2015 12:55

Ok, obrigado Ronaldo.

Reginaldo Cardoso
Reginaldo Cardoso
01/11/2016 15:41

Olá Vinicius, em nossa empresa estamos pensando em usar plataforma de Hardware aberta Linux e talvez precisaremos de contratar freelancer para ajudar nas interfaces com dispositivos externos…passe seu contato para eu te explicar melhor …Abs. Reginaldo ….reginaldo@trielo.com.br

VINICIUS FREIRE MACIEL
VINICIUS FREIRE MACIEL
Reply to  Reginaldo Cardoso
22/09/2018 17:58

Olá, desculpa, não estou recebendo notificações dos comentários.

Home » Software » Sistemas Operacionais » Comunicação SPI em Linux

EM DESTAQUE

WEBINARS

VEJA TAMBÉM

JUNTE-SE HOJE À COMUNIDADE EMBARCADOS

Talvez você goste: