mbed library sources

Dependents:   frdm_kl05z_gpio_test

Fork of mbed-src by mbed official

Committer:
mbed_official
Date:
Wed Mar 19 16:00:09 2014 +0000
Revision:
126:549ba18ddd81
Child:
227:7bd0639b8911
Synchronized with git revision cf8fd1cf86b0cd85131dd24a6ded21cc6fe04827

Full URL: https://github.com/mbedmicro/mbed/commit/cf8fd1cf86b0cd85131dd24a6ded21cc6fe04827/

Conflicts:
workspace_tools/targets.py

Who changed what in which revision?

UserRevisionLine numberNew contents of line
mbed_official 126:549ba18ddd81 1 /* mbed Microcontroller Library
mbed_official 126:549ba18ddd81 2 *******************************************************************************
mbed_official 126:549ba18ddd81 3 * Copyright (c) 2014, STMicroelectronics
mbed_official 126:549ba18ddd81 4 * All rights reserved.
mbed_official 126:549ba18ddd81 5 *
mbed_official 126:549ba18ddd81 6 * Redistribution and use in source and binary forms, with or without
mbed_official 126:549ba18ddd81 7 * modification, are permitted provided that the following conditions are met:
mbed_official 126:549ba18ddd81 8 *
mbed_official 126:549ba18ddd81 9 * 1. Redistributions of source code must retain the above copyright notice,
mbed_official 126:549ba18ddd81 10 * this list of conditions and the following disclaimer.
mbed_official 126:549ba18ddd81 11 * 2. Redistributions in binary form must reproduce the above copyright notice,
mbed_official 126:549ba18ddd81 12 * this list of conditions and the following disclaimer in the documentation
mbed_official 126:549ba18ddd81 13 * and/or other materials provided with the distribution.
mbed_official 126:549ba18ddd81 14 * 3. Neither the name of STMicroelectronics nor the names of its contributors
mbed_official 126:549ba18ddd81 15 * may be used to endorse or promote products derived from this software
mbed_official 126:549ba18ddd81 16 * without specific prior written permission.
mbed_official 126:549ba18ddd81 17 *
mbed_official 126:549ba18ddd81 18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
mbed_official 126:549ba18ddd81 19 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
mbed_official 126:549ba18ddd81 20 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
mbed_official 126:549ba18ddd81 21 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
mbed_official 126:549ba18ddd81 22 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
mbed_official 126:549ba18ddd81 23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
mbed_official 126:549ba18ddd81 24 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
mbed_official 126:549ba18ddd81 25 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
mbed_official 126:549ba18ddd81 26 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
mbed_official 126:549ba18ddd81 27 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
mbed_official 126:549ba18ddd81 28 *******************************************************************************
mbed_official 126:549ba18ddd81 29 */
mbed_official 126:549ba18ddd81 30 #include "spi_api.h"
mbed_official 126:549ba18ddd81 31
mbed_official 126:549ba18ddd81 32 #if DEVICE_SPI
mbed_official 126:549ba18ddd81 33
mbed_official 126:549ba18ddd81 34 #include <math.h>
mbed_official 126:549ba18ddd81 35 #include "cmsis.h"
mbed_official 126:549ba18ddd81 36 #include "pinmap.h"
mbed_official 126:549ba18ddd81 37 #include "error.h"
mbed_official 126:549ba18ddd81 38
mbed_official 126:549ba18ddd81 39 static const PinMap PinMap_SPI_MOSI[] = {
mbed_official 126:549ba18ddd81 40 {PA_7, SPI_1, STM_PIN_DATA(GPIO_Mode_AF_PP, 0)},
mbed_official 126:549ba18ddd81 41 {PB_5, SPI_1, STM_PIN_DATA(GPIO_Mode_AF_PP, 1)}, // Remap
mbed_official 126:549ba18ddd81 42 {NC, NC, 0}
mbed_official 126:549ba18ddd81 43 };
mbed_official 126:549ba18ddd81 44
mbed_official 126:549ba18ddd81 45 static const PinMap PinMap_SPI_MISO[] = {
mbed_official 126:549ba18ddd81 46 {PA_6, SPI_1, STM_PIN_DATA(GPIO_Mode_AF_PP, 0)},
mbed_official 126:549ba18ddd81 47 {PB_4, SPI_1, STM_PIN_DATA(GPIO_Mode_AF_PP, 1)}, // Remap
mbed_official 126:549ba18ddd81 48 {NC, NC, 0}
mbed_official 126:549ba18ddd81 49 };
mbed_official 126:549ba18ddd81 50
mbed_official 126:549ba18ddd81 51 static const PinMap PinMap_SPI_SCLK[] = {
mbed_official 126:549ba18ddd81 52 {PA_5, SPI_1, STM_PIN_DATA(GPIO_Mode_AF_PP, 0)},
mbed_official 126:549ba18ddd81 53 {PB_3, SPI_1, STM_PIN_DATA(GPIO_Mode_AF_PP, 1)}, // Remap
mbed_official 126:549ba18ddd81 54 {NC, NC, 0}
mbed_official 126:549ba18ddd81 55 };
mbed_official 126:549ba18ddd81 56
mbed_official 126:549ba18ddd81 57 // Only used in Slave mode
mbed_official 126:549ba18ddd81 58 static const PinMap PinMap_SPI_SSEL[] = {
mbed_official 126:549ba18ddd81 59 {PB_6, SPI_1, STM_PIN_DATA(GPIO_Mode_IN_FLOATING, 0)}, // Generic IO, not real H/W NSS pin
mbed_official 126:549ba18ddd81 60 {NC, NC, 0}
mbed_official 126:549ba18ddd81 61 };
mbed_official 126:549ba18ddd81 62
mbed_official 126:549ba18ddd81 63 static void init_spi(spi_t *obj) {
mbed_official 126:549ba18ddd81 64 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 65 SPI_InitTypeDef SPI_InitStructure;
mbed_official 126:549ba18ddd81 66
mbed_official 126:549ba18ddd81 67 SPI_Cmd(spi, DISABLE);
mbed_official 126:549ba18ddd81 68
mbed_official 126:549ba18ddd81 69 SPI_InitStructure.SPI_Mode = obj->mode;
mbed_official 126:549ba18ddd81 70 SPI_InitStructure.SPI_NSS = obj->nss;
mbed_official 126:549ba18ddd81 71 SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
mbed_official 126:549ba18ddd81 72 SPI_InitStructure.SPI_DataSize = obj->bits;
mbed_official 126:549ba18ddd81 73 SPI_InitStructure.SPI_CPOL = obj->cpol;
mbed_official 126:549ba18ddd81 74 SPI_InitStructure.SPI_CPHA = obj->cpha;
mbed_official 126:549ba18ddd81 75 SPI_InitStructure.SPI_BaudRatePrescaler = obj->br_presc;
mbed_official 126:549ba18ddd81 76 SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
mbed_official 126:549ba18ddd81 77 SPI_InitStructure.SPI_CRCPolynomial = 7;
mbed_official 126:549ba18ddd81 78 SPI_Init(spi, &SPI_InitStructure);
mbed_official 126:549ba18ddd81 79
mbed_official 126:549ba18ddd81 80 SPI_Cmd(spi, ENABLE);
mbed_official 126:549ba18ddd81 81 }
mbed_official 126:549ba18ddd81 82
mbed_official 126:549ba18ddd81 83 void spi_init(spi_t *obj, PinName mosi, PinName miso, PinName sclk, PinName ssel) {
mbed_official 126:549ba18ddd81 84 // Determine the SPI to use
mbed_official 126:549ba18ddd81 85 SPIName spi_mosi = (SPIName)pinmap_peripheral(mosi, PinMap_SPI_MOSI);
mbed_official 126:549ba18ddd81 86 SPIName spi_miso = (SPIName)pinmap_peripheral(miso, PinMap_SPI_MISO);
mbed_official 126:549ba18ddd81 87 SPIName spi_sclk = (SPIName)pinmap_peripheral(sclk, PinMap_SPI_SCLK);
mbed_official 126:549ba18ddd81 88 SPIName spi_ssel = (SPIName)pinmap_peripheral(ssel, PinMap_SPI_SSEL);
mbed_official 126:549ba18ddd81 89
mbed_official 126:549ba18ddd81 90 SPIName spi_data = (SPIName)pinmap_merge(spi_mosi, spi_miso);
mbed_official 126:549ba18ddd81 91 SPIName spi_cntl = (SPIName)pinmap_merge(spi_sclk, spi_ssel);
mbed_official 126:549ba18ddd81 92
mbed_official 126:549ba18ddd81 93 obj->spi = (SPIName)pinmap_merge(spi_data, spi_cntl);
mbed_official 126:549ba18ddd81 94
mbed_official 126:549ba18ddd81 95 if (obj->spi == (SPIName)NC) {
mbed_official 126:549ba18ddd81 96 error("SPI pinout mapping failed");
mbed_official 126:549ba18ddd81 97 }
mbed_official 126:549ba18ddd81 98
mbed_official 126:549ba18ddd81 99 // Enable SPI clock
mbed_official 126:549ba18ddd81 100 if (obj->spi == SPI_1) {
mbed_official 126:549ba18ddd81 101 RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);
mbed_official 126:549ba18ddd81 102 }
mbed_official 126:549ba18ddd81 103
mbed_official 126:549ba18ddd81 104 // Configure the SPI pins
mbed_official 126:549ba18ddd81 105 pinmap_pinout(mosi, PinMap_SPI_MOSI);
mbed_official 126:549ba18ddd81 106 pinmap_pinout(miso, PinMap_SPI_MISO);
mbed_official 126:549ba18ddd81 107 pinmap_pinout(sclk, PinMap_SPI_SCLK);
mbed_official 126:549ba18ddd81 108
mbed_official 126:549ba18ddd81 109 // Save new values
mbed_official 126:549ba18ddd81 110 obj->bits = SPI_DataSize_8b;
mbed_official 126:549ba18ddd81 111 obj->cpol = SPI_CPOL_Low;
mbed_official 126:549ba18ddd81 112 obj->cpha = SPI_CPHA_1Edge;
mbed_official 126:549ba18ddd81 113 obj->br_presc = SPI_BaudRatePrescaler_256; // 1MHz
mbed_official 126:549ba18ddd81 114
mbed_official 126:549ba18ddd81 115 if (ssel == NC) { // Master
mbed_official 126:549ba18ddd81 116 obj->mode = SPI_Mode_Master;
mbed_official 126:549ba18ddd81 117 obj->nss = SPI_NSS_Soft;
mbed_official 126:549ba18ddd81 118 }
mbed_official 126:549ba18ddd81 119 else { // Slave
mbed_official 126:549ba18ddd81 120 pinmap_pinout(ssel, PinMap_SPI_SSEL);
mbed_official 126:549ba18ddd81 121 obj->mode = SPI_Mode_Slave;
mbed_official 126:549ba18ddd81 122 obj->nss = SPI_NSS_Soft;
mbed_official 126:549ba18ddd81 123 }
mbed_official 126:549ba18ddd81 124
mbed_official 126:549ba18ddd81 125 init_spi(obj);
mbed_official 126:549ba18ddd81 126 }
mbed_official 126:549ba18ddd81 127
mbed_official 126:549ba18ddd81 128 void spi_free(spi_t *obj) {
mbed_official 126:549ba18ddd81 129 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 130 SPI_I2S_DeInit(spi);
mbed_official 126:549ba18ddd81 131 }
mbed_official 126:549ba18ddd81 132
mbed_official 126:549ba18ddd81 133 void spi_format(spi_t *obj, int bits, int mode, int slave) {
mbed_official 126:549ba18ddd81 134 // Save new values
mbed_official 126:549ba18ddd81 135 if (bits == 8) {
mbed_official 126:549ba18ddd81 136 obj->bits = SPI_DataSize_8b;
mbed_official 126:549ba18ddd81 137 }
mbed_official 126:549ba18ddd81 138 else {
mbed_official 126:549ba18ddd81 139 obj->bits = SPI_DataSize_16b;
mbed_official 126:549ba18ddd81 140 }
mbed_official 126:549ba18ddd81 141
mbed_official 126:549ba18ddd81 142 switch (mode) {
mbed_official 126:549ba18ddd81 143 case 0:
mbed_official 126:549ba18ddd81 144 obj->cpol = SPI_CPOL_Low;
mbed_official 126:549ba18ddd81 145 obj->cpha = SPI_CPHA_1Edge;
mbed_official 126:549ba18ddd81 146 break;
mbed_official 126:549ba18ddd81 147 case 1:
mbed_official 126:549ba18ddd81 148 obj->cpol = SPI_CPOL_Low;
mbed_official 126:549ba18ddd81 149 obj->cpha = SPI_CPHA_2Edge;
mbed_official 126:549ba18ddd81 150 break;
mbed_official 126:549ba18ddd81 151 case 2:
mbed_official 126:549ba18ddd81 152 obj->cpol = SPI_CPOL_High;
mbed_official 126:549ba18ddd81 153 obj->cpha = SPI_CPHA_1Edge;
mbed_official 126:549ba18ddd81 154 break;
mbed_official 126:549ba18ddd81 155 default:
mbed_official 126:549ba18ddd81 156 obj->cpol = SPI_CPOL_High;
mbed_official 126:549ba18ddd81 157 obj->cpha = SPI_CPHA_2Edge;
mbed_official 126:549ba18ddd81 158 break;
mbed_official 126:549ba18ddd81 159 }
mbed_official 126:549ba18ddd81 160
mbed_official 126:549ba18ddd81 161 if (slave == 0) {
mbed_official 126:549ba18ddd81 162 obj->mode = SPI_Mode_Master;
mbed_official 126:549ba18ddd81 163 obj->nss = SPI_NSS_Soft;
mbed_official 126:549ba18ddd81 164 }
mbed_official 126:549ba18ddd81 165 else {
mbed_official 126:549ba18ddd81 166 obj->mode = SPI_Mode_Slave;
mbed_official 126:549ba18ddd81 167 obj->nss = SPI_NSS_Hard;
mbed_official 126:549ba18ddd81 168 }
mbed_official 126:549ba18ddd81 169
mbed_official 126:549ba18ddd81 170 init_spi(obj);
mbed_official 126:549ba18ddd81 171 }
mbed_official 126:549ba18ddd81 172
mbed_official 126:549ba18ddd81 173 void spi_frequency(spi_t *obj, int hz) {
mbed_official 126:549ba18ddd81 174 // Choose the baud rate divisor (between 2 and 256)
mbed_official 126:549ba18ddd81 175 uint32_t divisor = SystemCoreClock / hz;
mbed_official 126:549ba18ddd81 176
mbed_official 126:549ba18ddd81 177 // Find the nearest power-of-2
mbed_official 126:549ba18ddd81 178 divisor = (divisor > 0 ? divisor-1 : 0);
mbed_official 126:549ba18ddd81 179 divisor |= divisor >> 1;
mbed_official 126:549ba18ddd81 180 divisor |= divisor >> 2;
mbed_official 126:549ba18ddd81 181 divisor |= divisor >> 4;
mbed_official 126:549ba18ddd81 182 divisor |= divisor >> 8;
mbed_official 126:549ba18ddd81 183 divisor |= divisor >> 16;
mbed_official 126:549ba18ddd81 184 divisor++;
mbed_official 126:549ba18ddd81 185
mbed_official 126:549ba18ddd81 186 uint32_t baud_rate = __builtin_ffs(divisor) - 2;
mbed_official 126:549ba18ddd81 187
mbed_official 126:549ba18ddd81 188 // Save new value
mbed_official 126:549ba18ddd81 189 obj->br_presc = ((baud_rate > 7) ? (7 << 3) : (baud_rate << 3));
mbed_official 126:549ba18ddd81 190
mbed_official 126:549ba18ddd81 191 init_spi(obj);
mbed_official 126:549ba18ddd81 192 }
mbed_official 126:549ba18ddd81 193
mbed_official 126:549ba18ddd81 194 static inline int ssp_readable(spi_t *obj) {
mbed_official 126:549ba18ddd81 195 int status;
mbed_official 126:549ba18ddd81 196 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 197 // Check if data is received
mbed_official 126:549ba18ddd81 198 status = ((SPI_I2S_GetFlagStatus(spi, SPI_I2S_FLAG_RXNE) != RESET) ? 1 : 0);
mbed_official 126:549ba18ddd81 199 return status;
mbed_official 126:549ba18ddd81 200 }
mbed_official 126:549ba18ddd81 201
mbed_official 126:549ba18ddd81 202 static inline int ssp_writeable(spi_t *obj) {
mbed_official 126:549ba18ddd81 203 int status;
mbed_official 126:549ba18ddd81 204 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 205 // Check if data is transmitted
mbed_official 126:549ba18ddd81 206 status = ((SPI_I2S_GetFlagStatus(spi, SPI_I2S_FLAG_TXE) != RESET) ? 1 : 0);
mbed_official 126:549ba18ddd81 207 return status;
mbed_official 126:549ba18ddd81 208 }
mbed_official 126:549ba18ddd81 209
mbed_official 126:549ba18ddd81 210 static inline void ssp_write(spi_t *obj, int value) {
mbed_official 126:549ba18ddd81 211 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 212 while (!ssp_writeable(obj));
mbed_official 126:549ba18ddd81 213 SPI_I2S_SendData(spi, (uint16_t)value);
mbed_official 126:549ba18ddd81 214 }
mbed_official 126:549ba18ddd81 215
mbed_official 126:549ba18ddd81 216 static inline int ssp_read(spi_t *obj) {
mbed_official 126:549ba18ddd81 217 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 218 while (!ssp_readable(obj));
mbed_official 126:549ba18ddd81 219 return (int)SPI_I2S_ReceiveData(spi);
mbed_official 126:549ba18ddd81 220 }
mbed_official 126:549ba18ddd81 221
mbed_official 126:549ba18ddd81 222 static inline int ssp_busy(spi_t *obj) {
mbed_official 126:549ba18ddd81 223 int status;
mbed_official 126:549ba18ddd81 224 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 225 status = ((SPI_I2S_GetFlagStatus(spi, SPI_I2S_FLAG_BSY) != RESET) ? 1 : 0);
mbed_official 126:549ba18ddd81 226 return status;
mbed_official 126:549ba18ddd81 227 }
mbed_official 126:549ba18ddd81 228
mbed_official 126:549ba18ddd81 229 int spi_master_write(spi_t *obj, int value) {
mbed_official 126:549ba18ddd81 230 ssp_write(obj, value);
mbed_official 126:549ba18ddd81 231 return ssp_read(obj);
mbed_official 126:549ba18ddd81 232 }
mbed_official 126:549ba18ddd81 233
mbed_official 126:549ba18ddd81 234 int spi_slave_receive(spi_t *obj) {
mbed_official 126:549ba18ddd81 235 return (ssp_readable(obj) && !ssp_busy(obj)) ? (1) : (0);
mbed_official 126:549ba18ddd81 236 };
mbed_official 126:549ba18ddd81 237
mbed_official 126:549ba18ddd81 238 int spi_slave_read(spi_t *obj) {
mbed_official 126:549ba18ddd81 239 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 240 return (int)SPI_I2S_ReceiveData(spi);
mbed_official 126:549ba18ddd81 241 }
mbed_official 126:549ba18ddd81 242
mbed_official 126:549ba18ddd81 243 void spi_slave_write(spi_t *obj, int value) {
mbed_official 126:549ba18ddd81 244 SPI_TypeDef *spi = (SPI_TypeDef *)(obj->spi);
mbed_official 126:549ba18ddd81 245 while (!ssp_writeable(obj));
mbed_official 126:549ba18ddd81 246 SPI_I2S_SendData(spi, (uint16_t)value);
mbed_official 126:549ba18ddd81 247 }
mbed_official 126:549ba18ddd81 248
mbed_official 126:549ba18ddd81 249 int spi_busy(spi_t *obj) {
mbed_official 126:549ba18ddd81 250 return ssp_busy(obj);
mbed_official 126:549ba18ddd81 251 }
mbed_official 126:549ba18ddd81 252
mbed_official 126:549ba18ddd81 253 #endif