From fe7bdb284e8f1a3011eafd3eddad599332b7d824 Mon Sep 17 00:00:00 2001 From: Gregory Nutt Date: Sun, 9 Nov 2014 07:15:48 -0600 Subject: [PATCH] EFM32: Fix issues associated with SPI bi order. From Pierre --- arch/arm/src/efm32/efm32_spi.c | 43 +++++++++++++++++++++++++--------- libc/syslog/lib_syslog.c | 2 +- 2 files changed, 33 insertions(+), 12 deletions(-) diff --git a/arch/arm/src/efm32/efm32_spi.c b/arch/arm/src/efm32/efm32_spi.c index 6a427281bb..d28d8ea202 100644 --- a/arch/arm/src/efm32/efm32_spi.c +++ b/arch/arm/src/efm32/efm32_spi.c @@ -159,6 +159,7 @@ struct efm32_spidev_s #endif uint8_t nbits; /* Width of word in bits (4-16) */ + bool lsbfirst; /* True: Bit order is LSB first */ bool initialized; /* True: Already initialized */ }; @@ -1007,32 +1008,50 @@ static void spi_setbits(struct spi_dev_s *dev, int nbits) const struct efm32_spiconfig_s *config; uint32_t regval; uint32_t setting; - unsigned int truebits; + bool lsbfirst; spivdbg("nbits=%d\n", nbits); DEBUGASSERT(priv && priv->config); config = priv->config; - /* Has the number of bits changed? */ + /* Bit order is encoded by the sign of nbits */ - if (nbits != priv->nbits) + if (nbits < 0) { + /* LSB first */ + + lsbfirst = true; + nbits = -nbits; + } + else + { + /* MSH first */ + + lsbfirst = false; + } + + /* Has the number of bits or the bit order changed? */ + + if (nbits != priv->nbits || lsbfirst != priv->lsbfirst) + { + /* Set the new bit order */ + regval = spi_getreg(config, EFM32_USART_CTRL_OFFSET); - if (nbits < 0) + if (lsbfirst) { regval &= ~USART_CTRL_MSBF; - truebits = -nbits; } else { regval |= USART_CTRL_MSBF; - truebits = nbits; } - spi_putreg(config, EFM32_USART_CLKDIV_OFFSET, regval); + spi_putreg(config, EFM32_USART_CTRL_OFFSET, regval); - switch (truebits) + /* Select the new number of bits */ + + switch (nbits) { case 4: setting = USART_FRAME_DATABITS_FOUR; @@ -1099,7 +1118,8 @@ static void spi_setbits(struct spi_dev_s *dev, int nbits) * faster */ - priv->nbits = nbits; + priv->nbits = nbits; + priv->lsbfirst = lsbfirst; } } @@ -1556,9 +1576,9 @@ static int spi_portinitialize(struct efm32_spidev_s *priv) USART_CTRL_CLKPHA_SAMPLELEADING; spi_putreg(config, EFM32_USART_CTRL_OFFSET, regval); - /* LSB First */ + /* MSB First */ - regval &= ~USART_CTRL_MSBF; + regval |= USART_CTRL_MSBF; spi_putreg(config, EFM32_USART_CTRL_OFFSET, regval); #ifndef CONFIG_SPI_OWNBUS @@ -1566,6 +1586,7 @@ static int spi_portinitialize(struct efm32_spidev_s *priv) priv->mode = SPIDEV_MODE0; #endif priv->nbits = 8; + priv->lsbfirst = false; /* 8 bits */ diff --git a/libc/syslog/lib_syslog.c b/libc/syslog/lib_syslog.c index 70e8e13de5..6364010461 100644 --- a/libc/syslog/lib_syslog.c +++ b/libc/syslog/lib_syslog.c @@ -198,7 +198,7 @@ int vsyslog(int priority, FAR const char *fmt, va_list ap) if ((g_syslog_mask & LOG_MASK(priority)) != 0) { - /* Yes.. let vsylog_internal to the deed */ + /* Yes.. let vsylog_internal do the deed */ ret = vsyslog_internal(fmt, ap); }