### Fixes to Lab 4 - The wheel has 15 holes, not 19. - $\bullet$ The minimum speed you should be able to measure should be 20 RPM, not 5 RPM. ### The 9S12 Serial Peripheral Interface (SPI) - The 9S12 has a Synchronous Serial Interface - On the 9S12 it is called the Serial Peripheral Interface (SPI) - Information on the SPI can be found in the SPI Block User Guide. - If an 9S12 generates the clock used for the synchronous data transfer it is operating in Master Mode. - If an 9S12 uses and external clock used for the synchronous data transfer it is operating in Slave Mode. - If two 9S12's talk to each other using their SPI's one must be set up as the Master and the other as the Slave. - The output of the Master SPI shift register is connected to the input of the Slave SPI shift register over the Master Out Slave In (MOSI) line. - The input of the Master SPI shift register is connected to the output of the Slave SPI shift register over the Master In Slave Out (MISO) line. - After 8 clock ticks, the data originally in the Master shift register has been transferred to the slave, and the data in the Slave shift register has been transferred to the Master. # **Synchronous Serial Communications** #### Use of Slave Select with the 9S12 SPI - A master 9S12 can talk with more than one slave 9S12's - A slave 9S12 uses its Slave Select (SS) line to determine if it is the one the master is talking with - There can only be one master 9S12, because the master 9S12 is the device which generates the serial clock signal. # **Synchronous Serial Communications** With select lines, one master can communicate with more than one slave ### Using the 9S12 SPI with other devices - The 9S12 can communicate with many types of devices using its SPI - For example, consider a D/A (Digital-to-Analog) Converter - The D/A converter has three digital lines connected to the 9S12: - Serial Data - Serial Clock - Chip Select - The 9S12 can send a digital number to the D/A converter. The D/A converter will convert this digital number to a voltage. # SPI Communication with a D/A Converter #### Using the 9S12 SPI with other devices - Another type of device the 9S12 can talk to is a Real Time Clock (RTC) - An RTC keeps track of the time (year, month, day, hour, minute, second) - An RTC can be programmed to generate an alarm (interrupt) at a particular time (07:00), or can generate a periodic interrupt at a regular interval (once a second, once an hour, etc.) - The 9S12 initially tells the RTC what the correct time is. - The RTC keeps track of time from then on. ### **SPI Communication with a Real Time Clock** • In a system, an 9S12 can communicate with many different devices over its SPI interface. # Using the HCS12 SPI with other devices • An interface with even fewer wires can be implemented by using one data line in bidirectional mode. - In bidirectional mode, a single data line functions both as serial data in and serial data out. - In lab, we will connect our 9S12 to a Dallas Semiconductor DS1302 Real Time Clock, which uses a three-wire serial interface with a bidirectional data line. - The MOSI line on the 9S12 becomes a MOMI (Master Out Master In) line. - When the 9S12 wants to write data to the DS1302, it makes MIMO an output. - When the 9S12 wants to read data from the DS1302, it makes MIMO an input. # Bidirectional (3–Wire) SPI Communication with a Real Time Clock When used as a master in bidirectional mode, the Master Out Slave In pin becomes the Master Out Master In Pin • In a system, an HCS12 can communicate with many different devices over its SPI interface. • It uses the same data and clock lines, and selects different devices by using GPIO lines as slave selects. #### Using the 9S12 SPI • In synchronous serial communications, one device talks to another using a serial data line and a serial clock. - There are a number of decisions to be made before communication can begin. - For example - Is the 9S12 operating in master or slave mode? - Is the serial data sent out most significant bit (MSB) first, or least significant bit (LSB) first? - How many bits are sent in a single transfer cycle? - Is the data valid on the rising edge or the falling edge of the clock? - Is the data valid on the first edge or the second edge of the clock? - What is the speed of the data transfer (how many bits per second)? - Are there two uni-directional data lines or one bi-directional data line? - The 9S12 SPI is very versatile, and allows you to program all of these parameters. - The 9S12 SPI has 5 registers to set up and use the SPI system. | SPI0CR1 | SPIE | SPE | SPTIE | MSTR | CPOL | СРНА | SSOE | LSBFE | 0x00D8 | |---------|-------|-------|-------|--------|----------|-------|---------|-------|--------| | SPI0CR2 | 0 | 0 | 0 | MODFEN | BIRDIROE | 0 | SPISWAI | SPC0 | 0x00D9 | | SPI0BR | 0 | SPPR2 | SPPR1 | SPPR0 | 0 | SPR2 | SPR1 | SPR0 | 0x00DA | | SPI0SR | SPIF | 0 | SPTEF | MODF | 0 | 0 | 0 | 0 | 0x00DB | | SPI0DR | Bit 7 | Bit 6 | Bit 5 | Bit 4 | Bit 3 | Bit 2 | Bit 1 | Bit 0 | 0x00DD | ### Setting up the 9S12 SPI Clock Mode - You can program the SPI clock to determine the following things: - Is the data valid on the first or the second edge of the clock (clock phase)? - Is the clock idle high or idle low (clock polarity)? - This setup is done in the SPIOCRO register. ### Setting up the 9S12 SPI Clock Mode - The speed of the 9S12 clock is set up in the SPIOBR register. - The clock speed is set only if the 9S12 is being used as a master. - The possible clock speeds (for an 24 MHz E-clock) are: | SPPR2 | SPPR1 | SPPRO | SPR2 | SPR1 | SPR0 | E Clock | Frequency at | |-------|-------|-------|------|------|------|---------|--------------------| | | | | | | | Divisor | bus clock = 24 MHz | | 0 | 0 | 0 | 0 | 0 | 0 | 2 | 12.0 MHz | | 0 | 0 | 0 | 0 | 0 | 1 | 4 | 6.0 MHz | | 0 | 0 | 0 | 0 | 1 | 0 | 8 | 3.0 MHz | | 0 | 0 | 0 | 0 | 1 | 1 | 16 | 1.5 MHz | | 0 | 0 | 0 | 1 | 0 | 0 | 32 | 750.0 kHz | | 0 | 0 | 0 | 1 | 0 | 1 | 64 | 375.0 kHz | | 0 | 0 | 0 | 1 | 1 | 0 | 128 | 187.5 kHz | | 0 | 0 | 0 | 1 | 1 | 1 | 256 | 93.75 kHz | | 0 | 0 | 1 | 0 | 0 | 0 | 4 | 6.0 MHz | | 0 | 0 | 1 | 0 | 0 | 1 | 8 | 3.0 MHz | | 0 | 0 | 1 | 0 | 1 | 0 | 16 | 1.5 MHz | | 0 | 0 | 1 | 0 | 1 | 1 | 32 | 750.0 kHz | | 0 | 0 | 1 | 1 | 0 | 0 | 64 | 375.0 kHz | | 0 | 0 | 1 | 1 | 0 | 1 | 128 | 187.5 kHz | | 0 | 0 | 1 | 1 | 1 | 0 | 256 | 93.75 kHz | | 0 | 0 | 1 | 1 | 1 | 1 | 512 | 46.875 kHz | | 0 | 1 | 0 | 0 | 0 | 0 | 6 | 4.0 MHz | | 0 | 1 | 0 | 0 | 0 | 1 | 12 | 2.0 MHz | | 0 | 1 | 0 | 0 | 1 | 0 | 24 | 1.0 MHz | | 0 | 1 | 0 | 0 | 1 | 1 | 48 | 500.0 kHz | | 0 | 1 | 0 | 1 | 0 | 0 | 96 | 250.0 kHz | | 0 | 1 | 0 | 1 | 0 | 1 | 192 | 125.0 kHz | | 0 | 1 | 0 | 1 | 1 | 0 | 384 | 62.5 kHz | | 0 | 1 | 0 | 1 | 1 | 1 | 768 | 31.25 kHz | | SPPR2 | SPPR1 | SPPR0 | SPR2 | SPR1 | SPRO | E Clock | Frequency at | |-------|-------|-------|------|------|------|---------|--------------------| | | | | | | | Divisor | bus clock = 24 MHz | | 0 | 1 | 1 | 0 | 0 | 0 | 8 | 3.0 MHz | | 0 | 1 | 1 | 0 | 0 | 1 | 16 | 1.5 MHz | | 0 | 1 | 1 | 0 | 1 | 0 | 32 | 750.0 kHz | | 0 | 1 | 1 | 0 | 1 | 1 | 64 | 375.0 kHz | | 0 | 1 | 1 | 1 | 0 | 0 | 128 | 187.5 kHz | | 0 | 1 | 1 | 1 | 0 | 1 | 256 | 93.75 kHz | | 0 | 1 | 1 | 1 | 1 | 0 | 512 | 46.875 kHz | | 0 | 1 | 1 | 1 | 1 | 1 | 1024 | 23.438 kHz | | 1 | 0 | 0 | 0 | 0 | 0 | 10 | 2.4 MHz | | 1 | 0 | 0 | 0 | 0 | 1 | 20 | 1.2 MHz | | 1 | 0 | 0 | 0 | 1 | 0 | 40 | 600.0 kHz | | 1 | 0 | 0 | 0 | 1 | 1 | 80 | 300.0 kHz | | 1 | 0 | 0 | 1 | 0 | 0 | 160 | 150.0 kHz | | 1 | 0 | 0 | 1 | 0 | 1 | 320 | 75.0 kHz | | 1 | 0 | 0 | 1 | 1 | 0 | 640 | 37.5 kHz | | 1 | 0 | 0 | 1 | 1 | 1 | 1280 | 18.75 kHz | | 1 | 0 | 1 | 0 | 0 | 0 | 12 | 2.0 MHz | | 1 | 0 | 1 | 0 | 0 | 1 | 24 | 1.0 MHz | | 1 | 0 | 1 | 0 | 1 | 0 | 48 | 500.0 kHz | | 1 | 0 | 1 | 0 | 1 | 1 | 96 | 250.0 kHz | | 1 | 0 | 1 | 1 | 0 | 0 | 192 | 125.0 kHz | | 1 | 0 | 1 | 1 | 0 | 1 | 384 | 62.5 kHz | | 1 | 0 | 1 | 1 | 1 | 0 | 768 | 31.25 kHz | | 1 | 0 | 1 | 1 | 1 | 1 | 1536 | 15.625 kHz | | 1 | 1 | 0 | 0 | 0 | 0 | 14 | 1.714 MHz | | 1 | 1 | 0 | 0 | 0 | 1 | 28 | 857.14 kHz | | 1 | 1 | 0 | 0 | 1 | 0 | 56 | 428.57 kHz | | 1 | 1 | 0 | 0 | 1 | 1 | 112 | 214.29 kHz | | 1 | 1 | 0 | 1 | 0 | 0 | 224 | 107.14 kHz | | 1 | 1 | 0 | 1 | 0 | 1 | 448 | 53.57 kHz | | 1 | 1 | 0 | 1 | 1 | 0 | 896 | 26.785 kHz | | 1 | 1 | 0 | 1 | 1 | 1 | 1792 | 13.39 kHz | | 1 | 1 | 1 | 0 | 0 | 0 | 16 | 1.5 MHz | | 1 | 1 | 1 | 0 | 0 | 1 | 32 | 750 kHz | | 1 | 1 | 1 | 0 | 1 | 0 | 64 | 375.0 kHz | | 1 | 1 | 1 | 0 | 1 | 1 | 128 | 187.5 kHz | | 1 | 1 | 1 | 1 | 0 | 0 | 256 | 93.75 kHz | | 1 | 1 | 1 | 1 | 0 | 1 | 512 | 46.875 kHz | | 1 | 1 | 1 | 1 | 1 | 0 | 1024 | 23.438 kHz | | 1 | 1 | 1 | 1 | 1 | 1 | 2048 | 11.719 kHz | ### Using the 9S12 Serial Peripheral Interface Things to set up when using the 9S12 SPI subsystem - Enable SPI - Master or Slave? - Master generates clock for data transfers; slave uses master's clock - MSB first or LSB first? - Normally, MSB first - Clock Polarity - Clock idle low or clock idle high? - Clock Phase - Data valid on first clock edge or second clock edge? - Clock Speed (set by Master) - Generate interrupt after data transfered? - Bidirectional Mode Use the following registers: SPI0CR1, SPI0CR2, SPI0BR, SPI0SR, SPI0DR - 1. Enable SPI (SPE bit of SPIOCR1) - 2. Clock phase and polarity set to match device communicating with - 3. Select clock polarity CPOL bit of SPIOCR1 - CPOL = 0 for clock idle low - CPOL = 1 for clock idle high - 4. Select clock phase CPHA bit of SPIOCR1 - CPHA = 0 for data valid on first clock edge - CPHA = 1 for data valid on second clock edge - 5. Select master or slave MSTR bit of SPIOCR1 - Will be master when talking to devices such as D/A, A/D, clock, etc. - May be slave if talking to another microprocessor - 6. If you want to receive interrupt after one byte transferred, enable interrupts with SPIE bit of SPIOCR1 - Normally master will not use interrupts transfers are fast enough that you will normally wait for transfer to complete - Will often use interrupts when configured as a slave you will get interrupt when master sends you data - 7. Configure LSBF of SPIOCR1 for MSB first (LSBF = 0) or LSB first (LSBF = 1) - For most devices, use MSB first - 8. Configure for uni-directional mode (bit SPC0 = 0) or bi-directiona mode (bit $\widehat{\mathrm{SPC0}} = 1$ ) in SPIOCR2 - Bidirectional mode (SPC0 = 1 in SPIOCR2) used for three-wire communication. - When in bidirectional mode, the BIDIROE bit of SPIOCR2 determines if the data line is input or output. receiver #### Master Mode: - 1. Set clock rate SPPR2:0 and SPR2:0 bits of SPIOBR - Normally select clock at highest rate compatible with slave - 2. If using bidirectional mode, MOSI pin is used for data (now called MOMI, or Master Out Master In). - 3. MISO automatically configured as input by choosing master mode - 4. Configure some way to select slave(s) probably SS if only one slave; other I/O bits if multiple slaves - 5. Start data transfer by writing byte to SPIODR - 6. After transfer complete (8 clock cycles), SPIF bit of SPIOSR set. - If writing data to slave, can send next byte to SPIODR - If reading data from slave, can read data from SPIODR - 7. Set up SSOE of SPIOCR1 - SSOE = 0 if you want to control SS yourself (to be able to send more than one byte with SS low) - SSOE = 1 and MODFEN = 1 if you want to SS controlled automatically (SS will be active for one byte at a time) #### Slave Mode: - 1. No need to set clock speed slave accepts data at rate sent by master (up to 12 MHz) - 2. If using bidirectional mode, MISO pin is used for data (now called SISO, or Slave In Slave Out). - 3. No need to Make MOSI, SCLK, and SS inputs this is done automatically when configuring 9S12 as slave - If receiving data from master, wait until SPIF flag of SPIOSR set (or until SPI interrupt received), then read data from SPIODR - If sending data to master, write data to SPIODR before master starts transfer ### A C program to use the 9S12 in master mode ``` #include "hcs12.h" main() { char tmp; /* Use Bit 0 of Port A for Slave Select */ DDRA = Oxff; /* Port A output */ * SPI Setup SPIOCR1 = 0x50; /* 0 1 0 1 0 0 0 |\ |\ |\ |\ |\ \setminus_{---} MSB first | | | | \____ Do not use SS to automatically select slave | | | | \_____ O polarity (clock idle low) | \ | \ | No interrupts on transmit | \_____ Enable SPI _____ No interrupts */ SPIOCR2 = 0x00; /* Normal (not bi-directional) mode */ /* 12 MHz SPI clock */ SPIOBR = 0x00; /*********************** * End of SPI Setup PORTA = PORTA \& ~OxO1; /* Select slave */ while ((SPIOSR & Ox20) == 0); /* Wait for SPITEF flag */ SPIODR = 'h'; /* Send 'h' */ while ((SPIOSR & 0x80) == 0); /* Wait for transmission to complete */ /* Clear SPIF flag */ tmp = SPIODR; /* Deselect slave */ PORTA = PORTA \mid Ox01; PORTA = PORTA \& ~OxO1; while ((SPIOSR & Ox20) == 0); /* Wait for SPITEF flag */ SPIODR = 'e'; /* Send 'e' */ while ((SPIOSR & 0x80) == 0); /* Wait for transmission to complete */ tmp = SPIODR; /* Clear SPIF flag */ ``` ``` PORTA = PORTA \mid Ox01; PORTA = PORTA \& ~0x01; while ((SPIOSR & 0x20) == 0); /* Wait for SPITEF flag */ /* Send '1' */ SPIODR = '1'; while ((SPIOSR & 0x80) == 0); /* Wait for transmission to complete */ tmp = SPIODR; /* Clear SPIF flag */ PORTA = PORTA \mid Ox01; PORTA = PORTA & ~OxO1; while ((SPIOSR & 0x20) == 0); /* Wait for SPITEF flag */ /* Send '1' */ SPIODR = '1'; while ((SPIOSR & 0x80) == 0); /* Wait for transmission to complete */ tmp = SPIODR; /* Clear SPIF flag */ PORTA = PORTA \mid Ox01; PORTA = PORTA & ~OxO1; while ((SPIOSR & 0x20) == 0); /* Wait for SPITEF flag */ /* Send 'o' */ SPIODR = 'o'; while ((SPIOSR & 0x80) == 0); /* Wait for transfer to finish */ tmp = SPIODR; /* Clear SPIF flag */ PORTA = PORTA \mid Ox01; } ``` Here is a version using a loop to transfer data: ``` #include "hcs12.h" main() { const char data[] = "hello"; int i; char tmp; /* Use Bit 0 of Port A for Slave Select */ /* Port A output */ DDRA = Oxff; * SPI Setup SPIOCR1 = 0x50; /* 0 1 0 1 0 0 0 | | | | \ _____ Do not use SS to automatically select slave | | \_____ No interrupts on transmit | \_____ Enable SPI SPIOCR2 = 0x00; /* Normal (not bi-directional) mode */ SPIOBR = 0x00; /* 12 MHz SPI clock */ * End of SPI Setup for (i=0; ; i++) { if (data[i] == '\0') break; /* Exit loop at end of string */ PORTA = PORTA & ~OxO1; /* Select slave */ while ((SPIOSR & Ox20) == 0); /* Wait for SPITEF flag */ SPIODR = data[i]; /* Send data */ while ((SPIOSR & 0x80) == 0); /* Wait for transmission to complete */ /* Read SPIODR to clear SPIF */ tmp = SPIODR; /* Deselect slave */ PORTA = PORTA \mid Ox01; } } ```