

# AN712: Bringing Up New Microcontrollers for a Network Co-Processor and EmberZNet PRO 5.x

This document describes recommended practices and procedures for designing, developing, and testing the SPI Protocol on a new microcontroller that is interfaced to a Network Co-Processor (NCP) and using EmberZNet PRO 5.x stack software. The descriptions and recommendations illustrate a work flow that should help produce a solid SPI Protocol module for a new host microcontroller. **Caution:** This information applies only to EmberZNet PRO 5.x.

This document assumes that you have already read *AN711: SPI Host Interfacing Guide for ZigBee*. You should have an overall understanding of the SPI Protocol, the terminology used, the basic sequences described, and the transaction examples.

For debugging, a logic analyzer—preferably with at least seven channels—is highly recommended for examining the state of and the transitions on the NCP interface.

#### KEY FEATURES

- SPI Protocol pin connections and how to verify them.
- Example transaction instructions.
- · EZSP-SPI transaction details.
- Error condition descriptions.
- · Interfacing EZSP and SPI.
- Notes on using a Linux host.

# 1 Physical Interface

This section describes the SPI Protocol pin connections and how to verify them.

#### 1.1 SPI Protocol Pin Connections

The physical pin connections are straightforward, and there is only a special recommendation for the nHOST\_INT pin. nHOST\_INT can be connected to any input. For interrupt-based operation, nHOST\_INT must be connected to an external interrupt that can generate an interrupt on a falling edge. Furthermore, if the host intends to sleep and to be woken up by the NCP, nHOST\_INT should be connected to a pin that is capable of waking the host. nHOST\_INT should have a pull-up applied to it so that nHOST\_INT does not bounce in an unknown state if the NCP is reset. An internal pull-up on the pin that nHOST\_INT is connected to is acceptable.

Connect the three SPI signals (MOSI, MISO, and SCLK) to the host's SPI. Connect nSSEL to any output from the host that can operate Slave Select. For many microcontrollers, nSSEL will simply be connected to a general-purpose output. Connect nWAKE and nRESET to any general-purpose output from the host (remember, the NCP supplies an internal pull-up on both the nWAKE and nRESET pins).

## 1.2 Verifying SPI Protocol Pin Connections

Once all of the signals are connected and a logic analyzer is attached, begin by pulling the nRESET signal low for a short period (a minimum of 8 µs) to reset the NCP. nHOST\_INT will return to idle (go high) almost immediately after reset (if it is not already). Note that nHOST\_INT will not be driven high by a reset, but instead will default to an input. Therefore, if an external pull-up is not applied to nHOST\_INT, it is possible for nHOST\_INT to not go high immediately after reset but a short while later. During the startup sequence, the NCP will switch nHOST\_INT to an output and actively drive it high. After approximately 250 ms, the nHOST\_INT signal will assert (go low) and stay asserted until the host initiates a transaction. The startup time of the NCP can vary widely, but 250 ms is a good rule of thumb for when nHOST\_INT will assert after reset. nHOST\_INT asserting after pulling on the nRESET pin indicates that both the nRESET pin and nHOST\_INT are connected and operating correctly. If nHOST\_INT is tied to an external interrupt on the host, this is also a good time to test the interrupt generation by pulling on the nRESET pin to trigger nHOST\_INT assertion.

The simplest method to test nSSEL is to simply assert nSSEL (set low) and then deassert nSSEL. When nSSEL deasserts, nHOST\_INT will return to idle and then approximately 70 µs later will assert again and stay asserted.

Testing the three SPI signals (MOSI, MISO, and SCLK) is best done by formulating a complete transaction. Unfortunately, the nWAKE signal cannot be used or tested until a first, complete transaction has occurred (refer to the section **First Transaction: SPI Protocol Version**). This is because the nHOST\_INT signal must deassert after reset for a proper Wake Handshake to be performed. Once a complete transaction has finished and nHOST\_INT has deasserted, nWAKE may be asserted. Approximately 140 µs after nWAKE asserts, nHOST\_INT will assert in response, indicating that the nWAKE signal is connected properly.

#### 2 First Transaction: SPI Protocol Version

Obtaining the SPI Protocol Version is a compact, simplified, and special transaction that illustrates a full transaction. Being able to properly obtain the SPI Protocol Version not only verifies five of the seven interface pins (MOSI, MISO, SCLK, nSSEL, and nHOST\_INT), but it is also useful as a test for verifying that the NCP is active and that the SPI Protocol code being implemented on the host is compatible with the firmware on the NCP. Use this transaction to verify the connection with the NCP during the host's boot sequence.

Before worrying about creating a generic SPI Protocol module or handling error cases, it is best to get a feel for a transaction in the simplest way possible. The following steps should result in your first transaction and can be explicitly coded in a test function for reference.

#### 2.1 SPI Protocol Version Transaction with an NCP Reset

**Note:** The following steps begin by resetting the NCP to guarantee that it is in a known state. The NCP resetting is an error condition that results in the first transaction performed after a reset returning the reset error. These steps describe receiving this reset error instead of the SPI Protocol Version. *AN711: SPI Host Interfacing Guide for ZigBee*, discusses each data byte mentioned in the following steps.

- 1. Assert nRESET and release to reset the NCP and guarantee it is in a known state.
- 2. Wait for nHOST\_INT to assert, which indicates that the NCP is active.
- 3. Assert nSSEL to begin a transaction.
- 4. Transmit 0x0A.
- 5. Transmit 0xA7.
- 6. Continually transmit 0xFF until the byte received is not 0xFF. The first byte received that is not 0xFF will be 0x00.
- 7. Transmit 0xFF while receiving 0x02.
- 8. Transmit 0xFF while receiving 0xA7.
- 9. Deassert nSSEL to finish the transaction.

#### 2.2 SPI Protocol Version Transaction without an NCP Reset

- 1. Assert nSSEL to begin a transaction.
- 2. Transmit 0x0A.
- 3. Transmit 0xA7.
- 4. Continually transmit 0xFF until the byte received is not 0xFF. The first byte received that is not 0xFF will be 0x82.
- 5. Transmit 0xFF while receiving 0xA7.
- 6. Deassert nSSEL to finish the transaction.

# 3 Second Transaction: SPI Status

The SPI Status transaction is very similar to the SPI Protocol Version transaction (detailed in the previous section). Like the SPI Protocol Version transaction, this transaction provides a simple example of interaction with the NCP. Silicon Labs recommends this as a test transaction to verify the connection with the NCP during the host's boot sequence.

Note: AN711: SPI Host Interfacing Guide for ZigBee, discusses each data byte mentioned in the following steps.

## 3.1 SPI Status Transaction

- 1. Assert nSSEL to begin a transaction.
- 2. Transmit 0x0B.
- 3. Transmit 0xA7.
- 4. Continually transmit 0xFF until the byte received is not 0xFF. The first byte received that is not 0xFF will be 0xC1.
- 5. Transmit 0xFF while receiving 0xA7.
- 6. Deassert nSSEL to finish the transaction.

# 4 Performing a Hard Reset

When the host resets, it is far simpler to reset the NCP and begin from a known state as opposed to trying to recover and resync with the previous (unknown) state of the NCP. The recommended procedure when the host resets is to perform a Hard Reset of the NCP during bootup. A Hard Reset is defined as the following sequence:

- 1. Toggle nRESET (active low) to reset the NCP.
- Wait for nHOST\_INT to assert, which indicates that the NCP is active.
- 3. Perform a SPI Protocol Version transaction and verify that the Response from the NCP is the NCP Reset error condition.
- 4. Perform a SPI Protocol Version transaction and verify that the SPI Protocol Version number is as expected.
- 5. Perform a SPI Status transaction and verify that the NCP is "Alive."

The purpose of performing this Hard Reset on bootup is threefold.

- By guaranteeing that the NCP is freshly booted, just like the host, the host can proceed with standard node and network initialization instead of consuming extra code space just trying to determine what state the NCP was left in.
- Since the NCP generates the NCP Reset error, which will override any legitimate transaction Response, the Hard Reset can acknowledge this planned and expected error condition so that the EZSP or full application does not have to implement special handling. Therefore, whenever a NCP Reset error is experienced outside of a Hard Reset, it can be treated as a true unexpected error condition.
- The SPI Protocol Version and SPI Status transactions are specialized transactions not implemented or used by the normal EZSP. These transactions are intended to be utility devices that allow the host to perform a simple "handshake" with the NCP. This handshake not only verifies that the NCP is alive and available to the EZSP, but also that the SPI Protocol implemented in the NCP is compatible with the SPI Protocol implemented on the host.

## 5 Third Transaction: EZSP Version Command

Before implementing a generic SPI Protocol on the host, Silicon Labs recommends explicitly coding a third transaction for providing exposure to an EZSP Frame and the format of the data involved with an EZSP Frame. The EZSP Frame used in this transaction is the VERSION command. The VERSION command is required to be the first EZSP command issued to the NCP. It exercises the code path all the way through the NCP firmware. Therefore, this command is useful not only for verifying that the EZSP is active, but also for illustrating the implementation of an EZSP transaction.

Note: AN711: SPI Host Interfacing Guide for ZigBee, discusses each data byte mentioned in the following steps.

#### 5.1 EZSP Version Command Transaction

- 1. Assert nSSEL to begin a transaction.
- 2. Transmit 0xFE.
- 3. Transmit 0x04.
- 4. Transmit 0x00.
- Transmit 0x00.
- 6. Transmit 0x00.
- 7. Transmit 0x04.
- 8. Transmit 0xA7.
- 9. Continually transmit 0xFF until the byte received is not 0xFF. The first byte received that is not 0xFF will be 0xFE.
- 10. Transmit 0xFF while receiving 0x07.
- 11. Transmit 0xFF while receiving 0x00.
- 12. Transmit 0xFF while receiving 0x80.
- 13. Transmit 0xFF while receiving 0x00.
- 14. Transmit 0xFF while receiving 0x04.
- 15. Transmit 0xFF while receiving 0x02.
- 16. Transmit 0xFF while receiving 0x10. (Note that this value is the build number and may vary.)
- 17. Transmit 0xFF while receiving 0x45. (Note that for EmberZNet PRO versions 4.7 and higher the receive value is 0x47.)
- 18. Transmit 0xFF while receiving 0xA7.
- 19. Deassert nSSEL to finish the transaction.

## 6 The Wait Section

Transmitting a command is a basic operation that simply requires asserting Slave Select and "dumping" the command bytes on the SPI in the most convenient method available (such as using a for() loop over a manual write, an interrupt-driven write, or a DMA). Once the first byte of the Response is received and the transaction has moved into the Response Section, receiving a Response is a basic, three-step operation: decode the first two bytes to determine the length of the Response, receive that precise number of bytes, and then deassert Slave Select.

How the Wait Section is implemented and handled requires some careful consideration of the two techniques available: clock the SPI (also known as polling on the SPI or polling for data) and interrupt on the falling edge of nHOST\_INT.

#### 6.1 Clock the SPI (Polling for Data)

The simplest and most straightforward method for determining when a Response is ready is to continually clock the SPI until the NCP transmits a byte other than 0xFF. When the host "clocks the SPI," the host should simply transmit 0xFF, since transmitting 0xFF is considered an idle line. The major advantage of polling for data is that the simplicity of polling requires very little code space, and in most cases this can be implemented using either a while() or a do{} while loop. The disadvantage of polling for data is the blocking nature of polling. Since transactions must occur serially (meaning a transaction must complete before another transaction can begin), the blocking nature of polling for data is usually only an issue if the host needs to perform tasks not related to EmberZNet PRO.

For example, if the host captures a button press and must send a message over the network in response to the button press, blocking in a polling loop is not a critical issue because of the serial nature of the transaction. Conversely, if the host must periodically take an ADC measurement and perform calculations based on the measurement, then blocking in a polling loop might not be desirable.

Because the host is the SPI Master, there are essentially no timing requirements dictating when or how often the host should clock the SPI (the most important requirement is to keep transactions moving quickly so that messages do not back up in the NCP's buffers). Therefore, the host can clock the SPI at its convenience, which means that a developer can choose to implement the simplest solution possible and sit in a while() loop waiting for a response. The developer can also choose a more advanced solution: for the host to poll periodically for a response while allowing other tasks to execute on the host. Knowing that the Wait Sections of many transactions can be milliseconds long, the developer may decide to clock the SPI and check for a response only once every millisecond.

Silicon Labs recommends choosing the simplest solution possible in the context of the host's resources and other requirements. During development, starting with the simplest blocking while() loop is an easy solution that can be expanded and customized as development progresses.

#### 6.2 Interrupt on the Falling Edge of nHOST\_INT

The falling edge of the signal nHOST\_INT indicates that a Response is ready when the falling edge occurs while Slave Select is asserted. Instead of clocking the SPI (either by completely blocking or periodically polling) and waiting for a response, the host can be configured to interrupt on the falling edge of nHOST\_INT. Once the host sees a falling edge on nHOST\_INT, it must still clock the SPI until data other than 0xFF is received. The major advantage to interrupting on nHOST\_INT is the ability of the host to perform other tasks while waiting for a response. Remember, since a new transaction cannot begin until the previous transaction has completed, be careful not to accidentally overlap transactions. The major disadvantage to interrupting on nHOST\_INT is the potential for accidentally starting a new transaction before the previous transaction has completed.

Note: The host should not poll on the level of the nHOST\_INT signal. Despite nHOST\_INT remaining low until the host performs an action, only the falling edge of nHOST\_INT can be trusted to properly indicate data. The NCP will carefully schedule the falling edge of nHOST\_INT, but due to latency it cannot guarantee exactly when the nHOST\_INT signal will return to idle after the host performs an action.

# 7 Inter-Command Spacing

The inter-command spacing is a simple time requirement needed to guarantee that the NCP has finished processing a transaction and is ready to accept a new transaction. The inter-command spacing is currently defined as at least 1 ms between the rising edge of Slave Select (ending transaction) and the falling edge of Slave Select (starting transaction). If the host is capable of blocking for 1 ms, the simplest solution is to simply burn CPU cycles for 1ms after deasserting Slave Select. Since burning CPU cycles for 1 ms is often undesirable, Silicon Labs recommends using a simple timer. By setting or starting a timer when Slave Select is deasserted, the host can perform other tasks during the inter-command spacing. If a timer is used, the host must guarantee that any and all attempts at starting a new transaction are either blocked or stalled until the timer has expired. Once the timer has expired, the host may assert Slave Select and begin a new transaction.

# 8 Asynchronous Signaling

As noted in the section Interrupt on the Falling Edge of nHOST\_INT, this falling edge is used to indicate data availability. When nHOST\_INT is asserted outside of a transaction (Slave Select is idle), the assertion means there is asynchronous data waiting in the NCP for the host. Remember, the host should *not* poll on the level of the nHOST\_INT signal. Instead, the host should assign an interrupt to nHOST\_INT and use the falling edge (the interrupt) to set a flag or some similar marker. This way, the EZSP implementation on the host can regularly poll on the flag outside of the interrupt context and trigger the EZSP Callback command.

For more advanced functionality, you can connect nHOST\_INT to a pin that is capable of waking the host from sleep, and therefore enter a low power mode, while waiting for any incoming data, like a normal asynchronous callback.

**Note:** Care must be taken when enabling an interrupt on nHOST\_INT so that the proper piece of code is executed. nHOST\_INT is capable of indicating three different situations (wake, callback, and response), and these situations are best indicated by the current state of the nSSEL and nWAKE pins. Refer to *AN711: SPI Host Interfacing Guide for ZigBee*, for sample waveforms showing these signals together.

# 9 Waking the NCP

Waking the NCP should be a straightforward implementation that only requires you to choose between a polling or an interrupt mechanism for knowing when the NCP is ready (much like the rest of the SPI Protocol). After asserting the nWAKE signal, the host should either poll for a falling edge of nHOST\_INT or set up for an interrupt on the falling edge. As soon as the edge is seen, the host should deassert nWAKE and continue operating the EZSP as desired.

The only major caveat, as mentioned earlier about interrupting on nHOST\_INT, is to make sure the proper piece of code gets triggered in response and to not perform further EZSP operations inside of interrupt context.

## 10 Error Conditions

The error conditions encountered by the host are exactly that: errors. These errors are not meant to be encountered in a mature product and are primarily used as a development and debugging aid. If the host experiences an error condition, chances are the host and the NCP have become out of sync, and the code needed to recover would be exceptionally error prone. Therefore, it is reasonable for the host to treat all error conditions or timeouts in the same way as asserts, and simply reset both the host and the NCP.

There is one common exception to this rule: When the host *intentionally* resets the NCP (for example, as described in the section **Performing a Hard Reset**), the host must expect the NCP Reset error condition to occur on the next transaction. This error condition should be observed and discarded as expected and normal.

Note:

The application must be careful not to interfere with any operation that loads firmware onto the NCP (e.g., bootloading). The recommended practice is for the host to have access to and control of the NCP's nRESET signal, and to toggle nRESET if an error condition occurs. When the NCP is being loaded with new firmware, it will not be capable of responding to the host; the host may think the NCP is unresponsive and attempt to reset it, which will disrupt the loading of new firmware. You should consider the best method to avoid resetting the NCP in this situation. Some options include:

- Putting the application in some mode where it leaves the NCP alone.
- Holding the host in reset, bootloader, or some other innocuous mode.
- Disabling the host's access to the nRESET line on the NCP.
- Physically disconnecting nRESET.

## 10.1 Error Bytes

Four SPI Bytes indicate error conditions. When implementing the code to receive a Response from the NCP, the host must be capable of parsing the SPI Byte as soon as possible for any of these error conditions. The host must continue to receive the entire error before deasserting Slave Select and processing the error. With the exception of an *intentional* NCP Reset error condition, the host should report, through a printf or other simple method, these four errors to the developer for debugging purposes, but should ultimately result in an assert or similar reset mechanism.

## 10.2 Timeouts

There are only two timeouts the host can experience: Wait Section and Wake Handshake. Just like the Error Bytes, if either of these timeouts occurs, the application should report them to the developer for debugging purposes, but should ultimately result in an assert or similar reset mechanism.

The timeouts are best measured using a timer, but if necessary the host can simply burn a known amount of CPU cycles while waiting for either normal operation to resume or the limit of allowable CPU cycles. For the Wait Section Timeout, the time is measured from the end of the last byte transmitted in the Command to the start of the first byte received that is not 0xFF. For the Wake Handshake Timeout, the time is measured from the falling edge of nWAKE to the falling edge of nHOST\_INT.

# 11 Interfacing the EZSP and the SPI Protocol

Due to the serial nature of the EZSP (that is, transactions must occur in sequence instead of overlapping), Silicon Labs recommends that the EZSP interface into the SPI Protocol through a polling driven mechanism. For example, after calling a function <code>sendCom-mand()</code>, the EZSP could continually call a function <code>pollForResponse()</code>. Otherwise, the EZSP implementation should be carefully coded to prevent the host from accidentally overlapping transactions.

If the host's SPI Protocol is implemented using interrupts, the host should be careful to never perform a transaction inside of an interrupt context. This is especially important because a transaction or a wake handshake could require up to 300 ms, respectively.

# 12 Using a SPI Interface with a Linux Host

It is possible to use a SPI interface with a Linux based host. This requires a few manual changes to a normal Linux-based EZSP-generated host application.

By default, the Simplicity IDE selects an EZSP UART interface to communicate with the NCP. To enable SPI, you must remove the EZSP UART plugin from your project and replace it with the EZSP SPI plugin. You may do this from the Plugins tab in Simplicity Studio's Simplicity IDE.



Figure 1. Plugins Tab Detail

Save the project and generate its files. After generating the project, you must edit the generated board header file to configure the hardware settings for the project. These edits cannot be done through Simplicity Studio.

The board header file is located in the generated project folder and is named projectname \_board.h. The following macros can optionally be defined in this board header to specify the hardware configuration for the SPI driver. To preserve these changes, do not overwrite this file if you need to generate project files again.

Note: Any macro not defined in the board header file will revert to the defaults values as shown here.

| Macro                        | Default Value          |
|------------------------------|------------------------|
| #define NCP_SPI_DEVICE       | "/dev/spidev0.0"       |
| #define NCP_SPI_MODE         | 0                      |
| #define NCP_SPI_SPEED_HZ     | (1024 * 1024) // 1 MHz |
| #define NCP_CHIP_SELECT_GPIO | "8"                    |
| #define NCP_HOST_INT_GPIO    | "22"                   |
| #define NCP_RESET_GPIO       | "23"                   |
| #define NCP_WAKE_GPIO        | "24"                   |
| #define NCP_RESET_DELAY_US   | 26                     |





Products www.silabs.com/products



Quality www.silabs.com/quality



Support and Community community.silabs.com

## Disclaimer

Silicon Labs intends to provide customers with the latest, accurate, and in-depth documentation of all peripherals and modules available for system and software implementers using or intending to use the Silicon Labs products. Characterization data, available modules and peripherals, memory sizes and memory addresses refer to each specific device, and "Typical" parameters provided can and do vary in different applications. Application examples described herein are for illustrative purposes only. Silicon Labs reserves the right to make changes without further notice and limitation to product information, specifications, and descriptions herein, and does not give warranties as to the accuracy or completeness of the included information. Silicon Labs shall have no liability for the consequences of use of the information supplied herein. This document does not imply or express copyright licenses granted hereunder to design or fabricate any integrated circuits. The products are not designed or authorized to be used within any Life Support System without the specific written consent of Silicon Labs. A "Life Support System" is any product or system intended to support or sustain life and/or health, which, if it fails, can be reasonably expected to result in significant personal injury or death. Silicon Labs products are not designed or authorized for military applications. Silicon Labs products shall under no circumstances be used in weapons of mass destruction including (but not limited to) nuclear, biological or chemical weapons, or missiles capable of delivering such weapons.

#### **Trademark Information**

Silicon Laboratories Inc.®, Silicon Laboratories®, Silicon Labs®, SiLabs® and the Silicon Labs logo®, Bluegiga®, Bluegiga Logo®, Clockbuilder®, CMEMS®, DSPLL®, EFM®, EFM32®, EFR, Ember®, Energy Micro, Energy Micro logo and combinations thereof, "the world's most energy friendly microcontrollers", Ember®, EZLink®, EZRadio®, EZRadio®, EZRadioPRO®, Gecko®, ISOmodem®, Precision32®, ProSLIC®, Simplicity Studio®, SiPHY®, Telegesis, the Telegesis Logo®, USBXpress® and others are trademarks or registered trademarks of Silicon Labs. ARM, CORTEX, Cortex-M3 and THUMB are trademarks or registered trademarks of ARM Holdings. Keil is a registered trademark of ARM Limited. All other products or brand names mentioned herein are trademarks of their respective holders.



Silicon Laboratories Inc. 400 West Cesar Chavez Austin, TX 78701 USA