Skip to content

Latest commit

 

History

History
1817 lines (1196 loc) · 64.4 KB

adc.md

File metadata and controls

1817 lines (1196 loc) · 64.4 KB

Rust on RISC-V BL602: Is It Sunny?

📝 3 Aug 2021

Today we shall magically transform any RISC-V BL602 Board into a Light Sensor!

We'll do this two ways...

  1. First we code the firmware in C

    (By calling the BL602 IoT SDK)

  2. Then we port the C firmware to Rust with...

    Rust Wrapper for BL602 IoT SDK

    (New to Rust? No worries we have tips for you!)

Wait... Do all BL602 Boards have an onboard Light Sensor?

Nope, all we need is a BL602 Board with an LED!

Reading the LED with BL602's Analog-to-Digital Converter (ADC) will turn it into a simple, improvised Light Sensor.

Amazing! Will this work with any BL602 Board?

We have tested this with PineCone BL602 and its onboard LED.

It will probably work with any BL602 / BL604 Board with an onboard or external LED: Ai-Thinker Ai-WB2, PineDio Stack, Pinenut, DT-BL10, MagicHome BL602, ...

Will our Light Sensor detect any kind of light?

Our LED-turned-Light-Sensor works best for detecting sunlight... We'll learn why in a while.

(Yep It's Always Sunny in Singapore ... So this Sunlight Sensor won't be so useful in Singapore 😂)

Testing the improvised Light Sensor on PineCone BL602 RISC-V Board. BTW that's the moon

Testing the improvised Light Sensor on PineCone BL602 RISC-V Board. BTW that's the moon

BL602 ADC in C

On PineCone BL602, there's a Blue LED connected on GPIO Pin Number 11...

PineCone RGB LED Schematic

(From PineCone BL602 Schematic)

For light sensing, we shall read the voltage from this LED GPIO with BL602's Analog-to-Digital Converter (ADC).

(Because LEDs will produce a current when exposed to light. See this)

Let's study the C Firmware for BL602 ADC: sdk_app_adc2

By calling the BL602 ADC Low Level HAL (Hardware Abstraction Layer), we shall...

  1. Initialise the ADC Channel for reading our LED GPIO

  2. Compute the average value of the ADC Samples that have been read

Definitions

We start by defining the GPIO Pin Number that will be read via ADC: demo.c

/// GPIO Pin Number that will be configured as ADC Input.
/// PineCone Blue LED is connected on BL602 GPIO 11.
/// PineCone Green LED is connected on BL602 GPIO 14.
/// Only these GPIOs are supported: 4, 5, 6, 9, 10, 11, 12, 13, 14, 15
/// TODO: Change the GPIO Pin Number for your BL602 board
#define ADC_GPIO 11

Not all GPIOs are supported by BL602's ADC!

According to the BL602 Reference Manual, only the following GPIOs are supported for ADC: 4, 5, 6, 9, 10, 11, 12, 13, 14, 15

ADC GPIO Pin Numbers

Next we define the ADC Frequency. We shall read 10,000 ADC Samples every second...

/// We set the ADC Frequency to 10 kHz according to <https://wiki.analog.com/university/courses/electronics/electronics-lab-led-sensor?rev=1551786227>
/// This is 10,000 samples per second.
#define ADC_FREQUENCY 10000  //  Hz

For computing the average, we shall remember the last 1,000 ADC Samples read...

/// We shall read 1,000 ADC samples, which will take 0.1 seconds
#define ADC_SAMPLES 1000

Finally we set the ADC Gain to increase the sensitivity of the ADC...

/// Set ADC Gain to Level 1 to increase the ADC sensitivity.
/// To disable ADC Gain, set `ADC_GAIN1` and `ADC_GAIN2` to `ADC_PGA_GAIN_NONE`.
/// See <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/bl602/bl602_std/bl602_std/StdDriver/Inc/bl602_adc.h#L133-L144>
#define ADC_GAIN1 ADC_PGA_GAIN_1
#define ADC_GAIN2 ADC_PGA_GAIN_1

More about ADC Gain in a while.

Initialise the ADC Channel

Here's how we initialise the ADC Channel for reading our LED GPIO: demo.c

/// Command to init the ADC Channel and start reading the ADC Samples.
/// Based on `hal_adc_init` in <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/hal_drv/bl602_hal/hal_adc.c#L50-L102>
void init_adc(char *buf, int len, int argc, char **argv) {
  //  Only these GPIOs are supported: 4, 5, 6, 9, 10, 11, 12, 13, 14, 15
  assert(ADC_GPIO==4 || ADC_GPIO==5 || ADC_GPIO==6 || ADC_GPIO==9 || ADC_GPIO==10 || ADC_GPIO==11 || ADC_GPIO==12 || ADC_GPIO==13 || ADC_GPIO==14 || ADC_GPIO==15);

  //  For Single-Channel Conversion Mode, frequency must be between 500 and 16,000 Hz
  assert(ADC_FREQUENCY >= 500 && ADC_FREQUENCY <= 16000);

  //  Init the ADC Frequency for Single-Channel Conversion Mode
  int rc = bl_adc_freq_init(1, ADC_FREQUENCY);
  assert(rc == 0);

Our init_adc Command begins by validating the GPIO Pin Number and ADC Frequency.

Then it calls bl_adc_freq_init to set the ADC Frequency.

(Functions named bl_adc_* are defined in the BL602 ADC Low Level HAL)

The first parameter to bl_adc_freq_init selects the ADC Mode...

  • ADC Mode 0: Scan Conversion Mode

    BL602 ADC Controller reads One ADC Sample from Multiple ADC Channels.

    (So it's scanning across multiple ADC Channels, recording one sample per channel)

  • ADC Mode 1: Single-Channel Conversion Mode

    BL602 ADC Controller reads Multiple ADC Samples continuously from One ADC Channel.

    (This is the mode we're using)

Next we set the ADC GPIO Pin Number for ADC Mode 1 (Single-Channel Conversion)...

  //  Init the ADC GPIO for Single-Channel Conversion Mode
  rc = bl_adc_init(1, ADC_GPIO);
  assert(rc == 0);

To increase the ADC sensitivity, we set the ADC Gain...

  //  Enable ADC Gain to increase the ADC sensitivity
  rc = set_adc_gain(ADC_GAIN1, ADC_GAIN2);
  assert(rc == 0);

(More about this in a while)

BL602 ADC Controller shall transfer the ADC Samples directly into RAM, thanks to the Direct Memory Access (DMA) Controller...

  //  Init DMA for the ADC Channel for Single-Channel Conversion Mode
  rc = bl_adc_dma_init(1, ADC_SAMPLES);
  assert(rc == 0);

(First parameter of bl_adc_dma_init is the ADC Mode)

We configure the GPIO Pin for ADC Input...

  //  Configure the GPIO Pin as ADC Input, no pullup, no pulldown
  rc = bl_adc_gpio_init(ADC_GPIO);
  assert(rc == 0);

We set the DMA Context for the ADC Channel...

  //  Get the ADC Channel Number for the GPIO Pin
  int channel = bl_adc_get_channel_by_gpio(ADC_GPIO);

  //  Get the DMA Context for the ADC Channel
  adc_ctx_t *ctx = bl_dma_find_ctx_by_channel(ADC_DMA_CHANNEL);
  assert(ctx != NULL);

  //  Indicate that the GPIO has been configured for ADC
  ctx->chan_init_table |= (1 << channel);

(bl_dma_find_ctx_by_channel is defined in BL602 DMA HAL)

Finally we start the ADC Channel...

  //  Start reading the ADC via DMA
  bl_adc_start();
}

BL602 ADC Controller will read the ADC Samples continuously (from the GPIO Pin) into RAM (until we stop the ADC Channel).

Read the ADC Channel

After starting the ADC Channel, how do we fetch the ADC Samples that have been read?

Let's find out in demo.c ...

/// Command to compute the average value of the ADC Samples that have just been read.
/// Based on `hal_adc_get_data` in <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/hal_drv/bl602_hal/hal_adc.c#L142-L179>
void read_adc(char *buf, int len, int argc, char **argv) {
  //  Get the ADC Channel Number for the GPIO Pin
  int channel = bl_adc_get_channel_by_gpio(ADC_GPIO);
    
  //  Get the DMA Context for the ADC Channel
  adc_ctx_t *ctx = bl_dma_find_ctx_by_channel(ADC_DMA_CHANNEL);
  assert(ctx != NULL);

  //  Verify that the GPIO has been configured for ADC
  assert(((1 << channel) & ctx->chan_init_table) != 0);

Our read_adc Command begins by verifying the DMA Context for the ADC Channel.

Next we check whether the ADC Sampling has been completed for the ADC Channel...

  //  If ADC Sampling is not finished, try again later    
  if (ctx->channel_data == NULL) {
    printf("ADC Sampling not finished\r\n");
    return;
  }

Remember that the BL602 ADC Controller will read ADC Samples continuously and write the last 1,000 samples to RAM (via DMA).

Let's copy the last 1,000 ADC Samples from the DMA Context (in RAM) to a Static Array adc_data...

  //  Static array that will store 1,000 ADC Samples
  static uint32_t adc_data[ADC_SAMPLES];

  //  Copy the read ADC Samples to the static array
  memcpy(
    (uint8_t*) adc_data,             //  Destination
    (uint8_t*) (ctx->channel_data),  //  Source
    sizeof(adc_data)                 //  Size
  );  

Then we compute the average value of the ADC Samples in adc_data...

  //  Compute the average value of the ADC Samples
  uint32_t sum = 0;
  for (int i = 0; i < ADC_SAMPLES; i++) {
    //  Scale up the ADC Sample to the range 0 to 3199
    uint32_t scaled = ((adc_data[i] & 0xffff) * 3200) >> 16;
    sum += scaled;
  }
  printf("Average: %lu\r\n", (sum / ADC_SAMPLES));
}

The default ADC Configuration has roughly 12 Bits of Resolution per ADC Sample.

Thus we scale each ADC Sample to the range 0 to 3199.

And that's how we code BL602 ADC Firmware in C!

Running the BL602 ADC Firmware in C

Run the C Firmware

Watch what happens when we flash and run the C Firmware for BL602 ADC: sdk_app_adc2

  1. Enter this command to initialise the ADC Channel...

    init_adc
    

    (We've seen this function earlier)

  2. Place the BL602 Board (with LED) in a dark place.

  3. Enter the read_adc command a few times to get the average values of the last 1,000 ADC Samples...

    read_adc
      Average: 1416
    
    read_adc
      Average: 1416
    
    read_adc
      Average: 1416
    
  4. Now place the BL602 Board (with LED) under sunlight.

  5. Enter the read_adc command a few times...

    read_adc
      Average: 1408
    
    read_adc
      Average: 1408
    
    read_adc
      Average: 1408
    

    Note that the average values have dropped from 1416 to 1408.

  6. Place the BL602 Board (with LED) back in the dark and check the average values...

    read_adc
      Average: 1417
    
    read_adc
      Average: 1416
    
    read_adc
      Average: 1416
    

    The average values have increased from 1408 to 1416.

    Yep our improvised BL602 Light Sensor works!

Setting the ADC Gain

Set the ADC Gain

Let's chat about ADC Gain, which we used when reading the LED as a Light Sensor.

(ADC Gain probably won't be needed for reading most types of ADC Inputs)

Why do we need ADC Gain when reading an LED?

Our LED generates a tiny bit of current when exposed to light. To measure that tiny bit of current, we need to increase the ADC sensitivity.

Thus we increase the ADC Gain. (By default there's no ADC Gain)

BL602 HAL has a function that sets the ADC Gain right?

Sadly no. We need to go really low-level and call the BL602 Standard Driver for ADC.

(The BL602 Standard Driver directly manipulates the BL602 Hardware Registers)

Here's the low-level code that sets the ADC Gain: demo.c

/// Enable ADC Gain to increase the ADC sensitivity.
/// Based on ADC_Init in <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/bl602/bl602_std/bl602_std/StdDriver/Src/bl602_adc.c#L152-L230>
static int set_adc_gain(uint32_t gain1, uint32_t gain2) {
  //  Read the ADC Configuration Hardware Register
  uint32_t reg = BL_RD_REG(AON_BASE, AON_GPADC_REG_CONFIG2);

  //  Set the ADC Gain
  reg = BL_SET_REG_BITS_VAL(reg, AON_GPADC_PGA1_GAIN, gain1);
  reg = BL_SET_REG_BITS_VAL(reg, AON_GPADC_PGA2_GAIN, gain2);

  //  Set the ADC Chop Mode
  if (gain1 != ADC_PGA_GAIN_NONE || gain2 != ADC_PGA_GAIN_NONE) {
    reg = BL_SET_REG_BITS_VAL(reg, AON_GPADC_CHOP_MODE, 2);
  } else {
    reg = BL_SET_REG_BITS_VAL(reg, AON_GPADC_CHOP_MODE, 1);        
  }

  //  Enable the ADC PGA
  reg = BL_CLR_REG_BIT(reg, AON_GPADC_PGA_VCMI_EN);
  if (gain1 != ADC_PGA_GAIN_NONE || gain2 != ADC_PGA_GAIN_NONE) {
    reg = BL_SET_REG_BIT(reg, AON_GPADC_PGA_EN);
  } else {
    reg = BL_CLR_REG_BIT(reg, AON_GPADC_PGA_EN);
  }

  //  Update the ADC Configuration Hardware Register
  BL_WR_REG(AON_BASE, AON_GPADC_REG_CONFIG2, reg);
  return 0;
}

Create a BL602 Rust Project

Before diving into the Rust Firmware, let's walk through the steps for creating a BL602 Rust Project (like sdk_app_rust_adc)...

  1. Download the Source Code for BL602 IoT SDK...

    git clone --recursive https://github.com/lupyuen/bl_iot_sdk
  2. Copy the Project Folder for an existing Rust Project in bl_iot_sdk/customer_app, like sdk_app_rust_gpio ...

  3. Paste the Project Folder into bl_iot_sdk/customer_app and rename it (like sdk_app_rust_adc)...

    BL602 Rust Project

    Be sure to rename the Sub Folder too. (The sdk_app_rust_adc inside sdk_app_rust_adc)

    Delete the build_out folder if it exists.

  4. Edit the Makefile in the new folder and set the Project Name: sdk_app_rust_adc/Makefile

    ##  Set the project name
    PROJECT_NAME := sdk_app_rust_adc
    
  5. Set the GCC Compiler Options (if any) in the Makefile sdk_app_rust_adc / sdk_app_rust_adc / bouffalo.mk

  6. Edit the run.sh script in the new folder and set the Project Name: sdk_app_rust_adc/run.sh

    ##  Set the project name
    export APP_NAME=sdk_app_rust_adc
  7. Replace the Rust Source Code in sdk_app_rust_adc/ rust/src/lib.rs

  8. See the Appendix for the steps to define the Rust Commands for the Command-Line Interface in sdk_app_rust_adc / demo.c

  9. Remember to edit README.md and fill in the project details

BL602 ADC in Rust

Now we study the Rust Firmware for BL602 ADC: sdk_app_rust_adc

We have converted the C Firmware to Rust line by line, so the Rust code will look highly similar to C.

Recall that our firmware implements two commands...

  1. Initialise the ADC Channel for reading our LED GPIO

  2. Compute the average value of the ADC Samples that have been read

Here is the Rust implementation...

Definitions

We start by declaring to the Rust Compiler that we're calling the Rust Core Library (instead of Rust Standard Library): lib.rs

#![no_std]  //  Use the Rust Core Library instead of the Rust Standard Library, which is not compatible with embedded systems

(Rust Standard Library is too heavy for embedded programs)

Next we import the functions from Rust Core Library that will be used in a while...

//  Import Libraries
use core::{          //  Rust Core Library
  fmt::Write,        //  String Formatting    
  mem::transmute,    //  Pointer Casting
  panic::PanicInfo,  //  Panic Handler
};

We import the Rust Wrapper for BL602 IoT SDK...

use bl602_sdk::{     //  Rust Wrapper for BL602 IoT SDK
  adc,               //  ADC HAL
  dma,               //  DMA HAL
  puts,              //  Console Output
  Ptr,               //  C Pointer
  String,            //  Strings (limited to 64 chars)
};

We shall read GPIO 11 (the Blue LED) as ADC Input...

/// GPIO Pin Number that will be configured as ADC Input.
/// PineCone Blue LED is connected on BL602 GPIO 11.
/// PineCone Green LED is connected on BL602 GPIO 14.
/// Only these GPIOs are supported: 4, 5, 6, 9, 10, 11, 12, 13, 14, 15
/// TODO: Change the GPIO Pin Number for your BL602 board
const ADC_GPIO: i32 = 11;

BL602 ADC Controller shall read 10,000 ADC Samples per second, and remember the last 100 ADC Samples...

/// We set the ADC Frequency to 10 kHz according to <https://wiki.analog.com/university/courses/electronics/electronics-lab-led-sensor?rev=1551786227>
/// This is 10,000 samples per second.
const ADC_FREQUENCY: u32 = 10000;  //  Hz

/// We shall read 100 ADC samples, which will take 0.01 seconds
const ADC_SAMPLES: usize = 100;

(usize is similar to size_t in C, it's used to represent the size of arrays)

We shall set the ADC Gain to increase the ADC sensitivity...

/// Set ADC Gain to Level 1 to increase the ADC sensitivity.
/// To disable ADC Gain, set `ADC_GAIN1` and `ADC_GAIN2` to `ADC_PGA_GAIN_NONE`.
/// See <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/bl602/bl602_std/bl602_std/StdDriver/Inc/bl602_adc.h#L133-L144>
const ADC_GAIN1: u32 = ADC_PGA_GAIN_1;
const ADC_GAIN2: u32 = ADC_PGA_GAIN_1;

But ADC_PGA_GAIN_1 is missing from our Rust Wrapper.

Thus we copy the value from BL602 IoT SDK and define it here...

const ADC_PGA_GAIN_1: u32 = 1;  //  From <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/bl602/bl602_std/bl602_std/StdDriver/Inc/bl602_adc.h#L133-L144>

Rust Firmware for BL602 ADC

Initialise the ADC Channel

Here's our Rust Function init_adc that will be called by the BL602 Command-Line Interface: lib.rs

/// Command to init the ADC Channel and start reading the ADC Samples.
/// Based on `hal_adc_init` in <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/hal_drv/bl602_hal/hal_adc.c#L50-L102>
#[no_mangle]             //  Don't mangle the function name
extern "C" fn init_adc(  //  Declare `extern "C"` because it will be called by BL602 firmware
  _result: *mut u8,        //  Result to be returned to command-line interface (char *)
  _len:  i32,              //  Size of result buffer (int)
  _argc: i32,              //  Number of command line args (int)
  _argv: *const *const u8  //  Array of command line args (char **)
) {
  puts("[Rust] Init ADC");

(We won't be parsing the command-line arguments, so let's ignore the parameters passed to init_adc)

We start by validating the GPIO Pin Number and ADC Frequency...

  //  Only these GPIOs are supported: 4, 5, 6, 9, 10, 11, 12, 13, 14, 15
  assert!(ADC_GPIO==4 || ADC_GPIO==5 || ADC_GPIO==6 || ADC_GPIO==9 || ADC_GPIO==10 || ADC_GPIO==11 || ADC_GPIO==12 || ADC_GPIO==13 || ADC_GPIO==14 || ADC_GPIO==15);

  //  For Single-Channel Conversion Mode, frequency must be between 500 and 16,000 Hz
  assert!(ADC_FREQUENCY >= 500 && ADC_FREQUENCY <= 16000);

(Remember: Not all GPIOs are supported for ADC!)

Next we select ADC Mode 1 (Single-Channel Conversion) and set the ADC Frequency...

  //  Init the ADC Frequency for Single-Channel Conversion Mode
  adc::freq_init(1, ADC_FREQUENCY)
    .expect("ADC Freq failed");

We set the ADC GPIO Pin Number for ADC Mode 1...

  //  Init the ADC GPIO for Single-Channel Conversion Mode
  adc::init(1, ADC_GPIO)
    .expect("ADC Init failed");

To increase the ADC sensitivity, we set the ADC Gain...

  //  Enable ADC Gain to increase the ADC sensitivity
  let rc = unsafe { set_adc_gain(ADC_GAIN1, ADC_GAIN2) };  //  Unsafe because we are calling C function
  assert!(rc == 0);

(This calls our C function set_adc_gain, which shall be explained below)

BL602 ADC Controller shall transfer the ADC Samples directly into RAM, thanks to the Direct Memory Access (DMA) Controller...

  //  Init DMA for the ADC Channel for Single-Channel Conversion Mode
  adc::dma_init(1, ADC_SAMPLES as u32)
    .expect("DMA Init failed");

(First parameter of dma_init is the ADC Mode)

We configure the GPIO Pin for ADC Input...

  //  Configure the GPIO Pin as ADC Input, no pullup, no pulldown
  adc::gpio_init(ADC_GPIO)
    .expect("ADC GPIO failed");

And we fetch the DMA Context for the ADC Channel...

  //  Get the ADC Channel Number for the GPIO Pin
  let channel = adc::get_channel_by_gpio(ADC_GPIO)
    .expect("ADC Channel failed");

  //  Get the DMA Context for the ADC Channel
  let ptr = dma::find_ctx_by_channel(adc::ADC_DMA_CHANNEL as i32)
    .expect("DMA Ctx failed");

However the returned pointer ptr is actually a "void *" pointer from C.

To use the pointer in Rust, we cast it to a DMA Context Pointer...

  //  Cast the returned C Pointer (void *) to a DMA Context Pointer (adc_ctx *)
  let ctx = unsafe {     //  Unsafe because we are casting a pointer
    transmute::<         //  Cast the type...
      Ptr,               //  From C Pointer (void *)
      *mut adc::adc_ctx  //  To DMA Context Pointer (adc_ctx *)
    >(ptr)               //  For this pointer
  };

(More about transmute in the Appendix)

Now we may update the DMA Context for the ADC Channel...

  //  Indicate that the GPIO has been configured for ADC
  unsafe {  //  Unsafe because we are dereferencing a pointer
    (*ctx).chan_init_table |= 1 << channel;
  }

(We flag this as unsafe because we're dereferencing a pointer: ctx)

Finally we start the ADC Channel...

  //  Start reading the ADC via DMA
  adc::start()
    .expect("ADC Start failed");
}

BL602 ADC Controller will read the ADC Samples continuously (from the GPIO Pin) into RAM (until we stop the ADC Channel).

Rust Firmware to read BL602 ADC

Read the ADC Channel

Our ADC Channel has been started, how do we average the ADC Samples that have been read?

Let's check out the Rust Function read_adc in lib.rs ...

/// Command to compute the average value of the ADC Samples that have just been read.
/// Based on `hal_adc_get_data` in <https://github.com/lupyuen/bl_iot_sdk/blob/master/components/hal_drv/bl602_hal/hal_adc.c#L142-L179>
#[no_mangle]              //  Don't mangle the function name
extern "C" fn read_adc(   //  Declare `extern "C"` because it will be called by BL602 firmware
  _result: *mut u8,        //  Result to be returned to command-line interface (char *)
  _len:  i32,              //  Size of result buffer (int)
  _argc: i32,              //  Number of command line args (int)
  _argv: *const *const u8  //  Array of command line args (char **)
) {

First we fetch the DMA Context for the ADC Channel...

  //  Get the ADC Channel Number for the GPIO Pin
  let channel = adc::get_channel_by_gpio(ADC_GPIO)
    .expect("ADC Channel failed");
  
  //  Get the DMA Context for the ADC Channel
  let ptr = dma::find_ctx_by_channel(adc::ADC_DMA_CHANNEL as i32)
    .expect("DMA Ctx failed");

Again we cast the returned C pointer ptr to a DMA Context Pointer...

  //  Cast the returned C Pointer (void *) to a DMA Context Pointer (adc_ctx *)
  let ctx = unsafe {     //  Unsafe because we are casting a pointer
    transmute::<         //  Cast the type...
      Ptr,               //  From C Pointer (void *)
      *mut adc::adc_ctx  //  To DMA Context Pointer (adc_ctx *)
    >(ptr)               //  For this pointer
  };

(More about transmute in the Appendix)

Now we may verify the DMA Context for the ADC Channel...

  //  Verify that the GPIO has been configured for ADC
  unsafe {  //  Unsafe because we are dereferencing a pointer
    assert!(((1 << channel) & (*ctx).chan_init_table) != 0);
  }

(We flag this as unsafe because we're dereferencing a pointer: ctx)

And we check whether the ADC Sampling has been completed for the ADC Channel (channel_data shouldn't be null)...

  //  If ADC Sampling is not finished, try again later    
  if unsafe { (*ctx).channel_data.is_null() } {  //  Unsafe because we are dereferencing a pointer
    puts("ADC Sampling not finished");
    return;
  }

(Again we flag as unsafe because we're dereferencing the pointer ctx)

Remember that the BL602 ADC Controller will read ADC Samples continuously and write the last 100 samples to RAM (via DMA).

We define an array adc_data to store the last 100 samples temporarily (on the stack)...

  //  Array that will store the last 100 ADC Samples
  //  (`ADC_SAMPLES` is 100)
  let mut adc_data: [u32; ADC_SAMPLES]
    = [0; ADC_SAMPLES];  //  Init array to 100 zeroes

(Rust requires all variables to be initialised, so we set the array to 100 zeroes)

Let's copy the last 100 ADC Samples from the DMA Context (in RAM) to our array adc_data (on the stack)...

  //  Copy the read ADC Samples to the array
  unsafe {                    //  Unsafe because we are copying raw memory
    core::ptr::copy(          //  Copy the memory...
      (*ctx).channel_data,    //  From Source (ADC DMA data)
      adc_data.as_mut_ptr(),  //  To Destination (mutable pointer to adc_data)
      adc_data.len()          //  Number of Items (each item is uint32 or 4 bytes)
    );    
  }

(More about this in the Appendix)

(adc_data.len() returns the array length: 100)

Then we compute the average value of the ADC Samples in adc_data...

  //  Compute the average value of the ADC Samples
  let mut sum = 0;
  for i in 0..ADC_SAMPLES {  //  From 0 to 99, `..` excludes 100
    //  Scale up the ADC Sample to the range 0 to 3199
    let scaled = ((adc_data[i] & 0xffff) * 3200) >> 16;
    sum += scaled;
  }
  let avg = sum / ADC_SAMPLES as u32;

We scale each ADC Sample to the range 0 to 3199. (Because the default ADC Configuration produces 12-bit samples)

Finally we compose a formatted string with the average value and display it...

  //  Format the output
  let mut buf = String::new();
  write!(buf, "[Rust] Average: {}", avg)
    .expect("buf overflow");

  //  Display the formatted output
  puts(&buf);
}

(Yep Rust will helpfully check for buffer overflow... safer than sprintf!)

Default String Size is 64 characters, as defined in the BL602 Rust Wrapper.

(Similar to "char[64]" in C)

The formatted output will appear like so...

Output from Rust Firmware

And we're done... That's how we code BL602 ADC Firmware in Rust!

Build the BL602 Rust Firmware

We may download the Rust Firmware Binary File sdk_app_rust_adc.bin from...

Or follow these steps to build the Rust Firmware sdk_app_rust_adc.bin...

  1. Install rustup, blflash and xpack-riscv-none-embed-gcc

  2. Download the source code for the BL602 Rust Firmware...

    ## Download the master branch of lupyuen's bl_iot_sdk
    git clone --recursive --branch master https://github.com/lupyuen/bl_iot_sdk
    cd bl_iot_sdk/customer_app/sdk_app_rust_adc
  3. Edit the script run.sh in the sdk_app_rust_adc folder.

    This build script was created for macOS, but can be modified to run on Linux x64 and Windows WSL.

  4. In run.sh, set the following variables to the downloaded folders for blflash and xpack-riscv-none-embed-gcc...

    ##  Where blflash is located
    export BLFLASH_PATH=$PWD/../../../blflash
    
    ##  Where GCC is located
    export GCC_PATH=$PWD/../../../xpack-riscv-none-embed-gcc

    Save the changes into run.sh

  5. Build the firmware...

    ./run.sh
  6. We should see...

    ----- Building Rust app and BL602 firmware for riscv32imacf-unknown-none-elf / sdk_app_rust_adc...
    
    ----- Build BL602 Firmware
    + make
    ...
    LD build_out/sdk_app_rust_adc.elf
    ld: undefined reference to `init_adc'
    ld: undefined reference to `read_adc'
    ----- Ignore undefined references to Rust Library
    

    This means that the C code from our BL602 Firmware has been built successfully.

  7. Next the script compiles our Rust code into a static library: libapp.a

    ----- Build Rust Library
    + rustup default nightly
    
    + cargo build \
        --target ../riscv32imacf-unknown-none-elf.json \
        -Z build-std=core
    
    Updating crates.io index
    Compiling compiler_builtins v0.1.46
    Compiling core v0.0.0
    ...
    Compiling bl602-macros v0.0.2
    Compiling bl602-sdk v0.0.6
    Compiling app v0.0.1 (bl_iot_sdk/customer_app/sdk_app_rust_adc/rust)
    Finished dev [unoptimized + debuginfo] target(s) in 23.55s
    
  8. Finally the script links the Rust static library into our BL602 firmware...

    ----- Link BL602 Firmware with Rust Library
    + make
    use existing version.txt file
    LD build_out/sdk_app_rust_adc.elf
    Generating BIN File to build_out/sdk_app_rust_adc.bin
    ...
    Building Finish. To flash build output.
    

    Ignore the error from blflash, we'll fix this in a while.

  9. Our BL602 Rust Firmware file has been generated at...

    build_out/sdk_app_rust_adc.bin
    

    Let's flash this to BL602 and run it!

Check out the complete build log here...

More about the build script

More about the custom Rust target

Building the BL602 Rust Firmware

Flash the BL602 Rust Firmware

Here's how we flash the Rust Firmware file sdk_app_rust_adc.bin to BL602...

  1. Set BL602 to Flashing Mode and restart the board...

    For PineCone:

    • Set the PineCone Jumper (IO 8) to the H Position (Like this)

    • Press the Reset Button

    For BL10:

    • Connect BL10 to the USB port

    • Press and hold the D8 Button (GPIO 8)

    • Press and release the EN Button (Reset)

    • Release the D8 Button

    For Ai-Thinker Ai-WB2, Pinenut and MagicHome BL602:

    • Disconnect the board from the USB Port

    • Connect GPIO 8 to 3.3V

    • Reconnect the board to the USB port

  2. For macOS:

    Enter this at the command prompt...

    ./run.sh

    The script should automatically flash the firmware after building...

    ----- Flash BL602 Firmware
    
    + blflash flash build_out/sdk_app_rust_adc.bin \
        --port /dev/tty.usbserial-1410 \
        --initial-baud-rate 230400 \
        --baud-rate 230400
    
    Finished dev [unoptimized + debuginfo] target(s) in 0.61s
    Running `target/debug/blflash flash sdk_app_rust_adc.bin --port /dev/tty.usbserial-1420 --initial-baud-rate 230400 --baud-rate 230400`
    Start connection...
    5ms send count 115
    handshake sent elapsed 104.593µs
    Connection Succeed
    Bootrom version: 1
    Boot info: BootInfo { len: 14, bootrom_version: 1, otp_info: [0, 0, 0, 0, 3, 0, 0, 0, 61, 9d, c0, 5, b9, 18, 1d, 0] }
    Sending eflash_loader...
    Finished 1.595620342s 17.92KB/s
    5ms send count 115
    handshake sent elapsed 81.908µs
    Entered eflash_loader
    Skip segment addr: 0 size: 47504 sha256 matches
    Skip segment addr: e000 size: 272 sha256 matches
    Skip segment addr: f000 size: 272 sha256 matches
    Erase flash addr: 10000 size: 135808
    Program flash... ed8a4cdacbc4c1543c74584d7297ad876b6731104856a10dff4166c123c6637d
    Program done 7.40735771s 17.91KB/s
    Skip segment addr: 1f8000 size: 5671 sha256 matches
    Success
    

    (We might need to edit the script to use the right serial port)

  3. For Linux and Windows:

    Copy build_out/sdk_app_rust_adc.bin to the blflash folder.

    Then enter this at the command prompt...

    ## TODO: Change this to the downloaded blflash folder
    cd blflash
    
    ## For Linux:
    blflash flash build_out/sdk_app_lora.bin \
        --port /dev/ttyUSB0
    
    ## For Windows: Change COM5 to the BL602 Serial Port
    blflash flash c:\blflash\sdk_app_lora.bin --port COM5

More details on flashing firmware

Running the BL602 Rust Firmware

Run the BL602 Rust Firmware

Finally we run the BL602 Rust Firmware...

  1. Set BL602 to Normal Mode (Non-Flashing) and restart the board...

    For PineCone:

    • Set the PineCone Jumper (IO 8) to the L Position (Like this)

    • Press the Reset Button

    For BL10:

    • Press and release the EN Button (Reset)

    For Ai-Thinker Ai-WB2, Pinenut and MagicHome BL602:

    • Disconnect the board from the USB Port

    • Connect GPIO 8 to GND

    • Reconnect the board to the USB port

  2. For macOS:

    The run.sh script should automatically launch CoolTerm after flashing...

    ----- Run BL602 Firmware
    + open -a CoolTerm
    

    More about CoolTerm

    For Linux:

    Connect to BL602's UART Port at 2 Mbps like so...

    screen /dev/ttyUSB0 2000000

    For Windows:

    Use putty (See this)

    Alternatively:

    Use the Web Serial Terminal (See this)

    More details on connecting to BL602

  3. In the serial console, enter the init_adc command to initialise the ADC Channel...

    init_adc
      [Rust] Init ADC
    

    (We've seen this function earlier)

  4. Place the BL602 Board (with LED) in a dark place.

  5. Enter the read_adc command a few times to get the average values of the last 1,000 ADC Samples...

    read_adc
      [Rust] Average: 1417
    read_adc
      [Rust] Average: 1417
    read_adc
      [Rust] Average: 1417
    
  6. Now place the BL602 Board (with LED) under sunlight.

  7. Enter the read_adc command a few times...

    read_adc
      [Rust] Average: 1411
    read_adc
      [Rust] Average: 1411
    read_adc
      [Rust] Average: 1412
    

    Note that the average values have dropped from 1417 to 1412.

  8. Place the BL602 Board (with LED) back in the dark and check the average values...

    read_adc
      [Rust] Average: 1417
    read_adc
      [Rust] Average: 1417
    read_adc
      [Rust] Average: 1417
    

    The average values have increased from 1412 to 1417.

    Our improvised BL602 Light Sensor works in Rust yay!

From C To Rust

I'm new to Rust. Is there an easier way to jump from C to Rust?

Today we've seen that it's feasible to translate C Firmware into Rust line by line...

Compare C and Rust

Which is great for embedded developers new to Rust!

Just be mindful of the differences between C and Rust...

  1. BL602 HAL Functions have been renamed for Rust.

    (Like "bl_adc_init" becomes "adc::init")

    To see the list of BL602 HAL Functions for Rust, check out the bl602-sdk documentation.

    (More about this in the next chapter)

  2. In Rust we check for BL602 HAL Errors by calling "expect" instead of "assert".

    (Rust Compiler will warn us if we forget to "expect")

  3. Rust is super strict about Mutability... Only variables and pointers declared "mut" can be changed.

    (That's why we write "*mut i32" to get a pointer to an integer whose value may be changed)

  4. Pointer Deferencing like "ptr->field" doesn't work in Rust.

    We rewrite it in Rust as "(*ptr).field"

  5. Rust will helpfully check for Buffer Overflow.

    (No more silent "sprintf" overflow!)

    For BL602 Rust Wrapper the default string size is 64 characters.

    (Similar to "char[64]" in C)

  6. All Rust variables shall be initialised before use.

    (Even arrays and structs!)

Let's talk about "unsafe" code in Rust...

Safer Rust

Rust reminds us to be Extra Careful when we work with C Functions and C Pointers.

That's why we need to flag the following code as unsafe...

  1. Calling C Functions

    //  Call the C function `set_adc_gain`
    unsafe { set_adc_gain(ADC_GAIN1, ADC_GAIN2) };

    (More about this in the Appendix)

  2. Casting C Pointers to Rust

    //  Cast a C Pointer to a Rust Pointer
    let ctx = unsafe {
      transmute::<         //  Cast the type...
        Ptr,               //  From C Pointer (void *)
        *mut adc::adc_ctx  //  To DMA Context Pointer (adc_ctx *)
      >(ptr)               //  For this pointer
    };

    (More about this in the Appendix)

  3. Dereferencing C Pointers

    //  Dereference a C Pointer (ctx)
    unsafe {
      (*ctx).chan_init_table = ...
    }
  4. Copying Memory with C Pointers

    //  Copy memory with a C Pointer (channel_data)
    unsafe {
      core::ptr::copy(          //  Copy the memory...
        (*ctx).channel_data,    //  From Source (ADC DMA data)
        adc_data.as_mut_ptr(),  //  To Destination (mutable pointer to adc_data)
        adc_data.len()          //  Number of Items (each item is uint32 or 4 bytes)
      );    
    }

    (More about this in the Appendix)

Accessing Static Variables is also "unsafe". Let's talk about this...

Static Variables in Rust

Earlier we saw this Rust code for averaging the ADC Samples...

//  `adc_data` will store 100 ADC Samples (`ADC_SAMPLES` is 100)
let mut adc_data: [u32; ADC_SAMPLES] = [0; ADC_SAMPLES];

//  Omitted: Copy data into `adc_data`
...

//  Compute average of `adc_data`
for i in 0..ADC_SAMPLES {
  //  Get value from `adc_data`
  let scaled = adc_data[i] & ...

Note that adc_data lives on the stack.

That's a huge chunk of data on the stack... 400 bytes!

What if we turn adc_data into a Static Array?

We convert adc_data to a Static Array like this...

//  `adc_data` becomes a Static Array
static mut adc_data: [u32; ADC_SAMPLES] = [0; ADC_SAMPLES];

adc_data no longer lives on the stack, it's now in Static Memory.

What's the catch?

Unfortunately Static Variables in Rust are unsafe.

Thus all references to adc_data must be flagged as unsafe...

//  `adc_data` is now unsafe because it's a Static Variable
let scaled = unsafe { adc_data[i] } & ...

Which makes the code harder to read. That's why we left adc_data on the stack for this tutorial.

Why are Static Variables unsafe?

Because it's potentially possible to execute the above code in multiple tasks...

Which produces undefined behaviour when multiple tasks access the same Static Variable.

So it's perfectly OK to use Static Variables in Rust. Just that we need to...

  1. Flag the Static Variables as unsafe

  2. Ensure ourselves that Static Variables are only accessed by one task at a time

Rust Wrapper for BL602 IoT SDK

Rust Wrapper for BL602 IoT SDK

The BL602 Rust Wrapper Functions look mighty similar to the C Functions from the BL602 IoT SDK. How is this possible?

Because the Rust Functions were automatically generated from BL602 IoT SDK!

We ran a script to generate the Rust Wrapper for BL602 IoT SDK.

And we published the Rust Wrapper on crates.io...

Which functions from the BL602 IoT SDK are supported?

Today our BL602 Rust Wrapper supports...

ADC I2C UART
DMA PWM WiFi
GPIO SPI

(See the complete list)

How do we add the BL602 Rust Wrapper to our Rust Project?

Just add bl602-sdk to the Rust project configuration: rust/Cargo.toml

## External Rust libraries used by this module.  See crates.io.
[dependencies]
bl602-sdk = "0.0.6"  # Rust Wrapper for BL602 IoT SDK: https://crates.io/crates/bl602-sdk

(Change "0.0.6" to the latest version on crates.io)

The BL602 Rust Wrapper will be auto-downloaded from crates.io when building the project.

BL602 Rust Wrapper Documentation

Is the BL602 Rust Wrapper documented?

Yep! Every Rust Function is linked to the section in "The RISC-V BL602 Book" that explains how we call the function...

(Check the Appendix to learn more about the BL602 Rust Wrapper)

Here's a sample project that calls the Rust Wrapper for GPIO...

Rust Wrapper for GPIO

(Source)

Why Sunlight?

Why does our BL602 LED detect only sunlight? And not other kinds of light?

We're guessing because...

  1. Sunlight is more intense

    (And produces more current)

  2. We used the Blue LED on PineCone BL602

    (Which is sensitive to the spectrum of Blue - Indigo - Violet - Ultra-Violet (UV) light)

LED Light Sensitivity is explained in this article...

As a photodiode, an LED is sensitive to wavelengths equal to or shorter than the predominant wavelength it emits. A green LED would be sensitive to blue light and to some green light, but not to yellow or red light.

Also according to Sravan Senthilnathan on Twitter...

This is possible due to the blue LED's semiconductor bandgap being one appropriate to absorb the UV spectrum sun light. Here is more info about it: Aluminium gallium nitride on Wikipedia

And according to Dan Lafleur on LinkedIn...

The technique I used to measure light level with an LED was to reverse bias the LED and then time the discharge. The light level is time based. The higher the light level the longer it takes to discharge the reversed bias junction.

(More about measuring the Discharge Duration of an LED)

Here's another comment...

Yes, I have used LEDs as light sensors in class projects. The point is that they are not just emitters. A selection of red, green and blue LEDs can be used to build a crude colour detector. It works.

That sounds interesting...

PineCone BL602 has Red, Green and Blue LEDs. Can we use the other LEDs?

PineCone RGB LED Schematic

(From PineCone BL602 Schematic)

Unfortunately PineCone's Red LED is connected on GPIO 17, which is not supported for ADC.

But PineCone's Green LED (GPIO 14) should work OK with ADC.

Exercise For The Reader: Use PineCone's Green LED (instead of the Blue LED) as an ADC Light Sensor. What kind of light does it detect?

(This article was inspired by the BBC micro:bit, which uses LED as a Light Sensor. See this)

Testing the improvised Light Sensor on PineCone BL602 with Pinebook Pro

Testing the improvised Light Sensor on PineCone BL602 with Pinebook Pro

What's Next

Today we've seen that it's viable to create Rust Firmware for BL602... Just call the Rust Wrapper for BL602 IoT SDK!

Soon we shall test the Rust Firmware on PineDio Stack BL604 with LoRa SX1262... As we explore whether it's feasible to teach Rust as a Safer Way to create firmware for BL602 and BL604.

Also we might Simulate BL602 Rust Firmware in a Web Browser with WebAssembly!

Many Thanks to my GitHub Sponsors for supporting my work! This article wouldn't have been possible without your support.

Got a question, comment or suggestion? Create an Issue or submit a Pull Request here...

lupyuen.github.io/src/adc.md

Notes

  1. This article is the expanded version of this Twitter Thread on Rust Wrapper for BL602 IoT SDK

    And this Twitter Thread on BL602 ADC

  2. Are there other ways to run Rust Firmware on BL602? See this...

    "Rust On BL602: Two More Ways"

  3. We may also use BL602 ADC HAL to read the BL602 Internal Temperature Sensor...

    "Internal Temperature Sensor on BL602"

  4. Is there a simpler way to code ADC Firmware in C?

    Yes, we could call the ADC High Level HAL.

    (Instead of the ADC Low Level HAL that we've seen)

    Here's our ADC Firmware, rewritten to call the ADC High Level HAL...

    BL602 ADC High Level HAL

    (Source)

    But the ADC High Level HAL won't let us set the ADC Gain.

    We need to patch the ADC Low Level HAL like so...

    Setting the ADC Gain by patching the ADC High Level HAL

    (Source)

    Also note that the ADC High Level HAL doesn't allow us to compute the average of the ADC Samples.

    It returns only one ADC Sample.

  5. ESP32 has something similar to the BL602 Rust Wrapper...

    (Perhaps someday we might wrap the BL602 Rust Wrapper into a Rust Embedded HAL for BL602 / BL604)

Appendix: Call C Functions from Rust

How do we call our own C Functions from Rust?

Earlier we saw this Rust code that sets the ADC Gain: lib.rs

//  In Rust: Enable ADC Gain to increase the ADC sensitivity
unsafe { set_adc_gain(ADC_GAIN1, ADC_GAIN2) };  //  Unsafe because we are calling C function

This calls the C Function set_adc_gain, which we have imported into Rust here: lib.rs

//  In Rust: Import C Function
extern "C" {
  /// Enable ADC Gain to increase the ADC sensitivity.
  /// Defined in customer_app/sdk_app_rust_adc/sdk_app_rust_adc/demo.c
  fn set_adc_gain(gain1: u32, gain2: u32) -> i32;
}

set_adc_gain is defined in this C source file: demo.c

What about the rest of the code in demo.c?

The rest of the C code in demo.c is needed to set up the Command-Line Interface for our BL602 Firmware.

We define the Rust Commands for the Command-Line Interface like so: demo.c

/// In C: Import Rust functions from customer_app/sdk_app_rust_adc/rust/src/lib.rs
void init_adc(char *buf, int len, int argc, char **argv);
void read_adc(char *buf, int len, int argc, char **argv);

/// List of commands. STATIC_CLI_CMD_ATTRIBUTE makes these commands static
const static struct cli_command cmds_user[] STATIC_CLI_CMD_ATTRIBUTE = {
    {"init_adc",        "Init ADC Channel",          init_adc},
    {"read_adc",        "Read ADC Channel",          read_adc},
};

This defines the commands init_adc and read_adc, which are mapped to the respective Rust Functions.

Appendix: Cast C Pointers in Rust

How do we cast C Pointers in Rust?

In the C version of our ADC Firmware, we implicitly cast a "void *" pointer to "adc_ctx *" pointer like this: demo.c

//  In C: Get the pointer (void *) for DMA Context
void *ptr = ...

//  Cast the returned pointer (void *) to a DMA Context Pointer (adc_ctx *)
struct adc_ctx *ctx = (struct adc_ctx *) ptr;

Here we're Downcasting a General Type (void *) to a Specific Type (adc_ctx *).

To do the same in Rust, we need to be super explicit about what we're casting: lib.rs

//  In Rust: Get the C Pointer (void *) for DMA Context
let ptr = ...

//  Cast the returned C Pointer (void *) to a DMA Context Pointer (adc_ctx *)
let ctx = unsafe {     //  Unsafe because we are casting a pointer
  transmute::<         //  Cast the type...
    Ptr,               //  From C Pointer (void *)
    *mut adc::adc_ctx  //  To DMA Context Pointer (adc_ctx *)
  >(ptr)               //  For this pointer
};

transmute is the Rust Core Library Function that will cast our value (ptr) from one type to another...

transmute::< FromType , ToType >( ptr )

Where...

  • From Type is Ptr

    Ptr is our short form for the "void *" pointer in Rust.

    (See this)

  • To Type is *mut adc::adc_ctx

    Which is a mutable pointer to adc_ctx

    (Equivalent to "adc_ctx *" in C)

(More about transmute)

Casting a C Pointer in Rust

Appendix: Copy Memory with C Pointers

How do we copy memory with C Pointers?

Earlier we saw this code in our C Firmware: demo.c

//  Array that will store ADC Samples
uint32_t adc_data[ADC_SAMPLES];

//  Copy the read ADC Samples to the array
memcpy(
  (uint8_t*) adc_data,             //  Destination
  (uint8_t*) (ctx->channel_data),  //  Source
  sizeof(adc_data)                 //  Size
);  

This code copies the ADC Samples from the DMA buffer to the array adc_data.

Here's the equivalent code in Rust: lib.rs

//  Array that will store the last 100 ADC Samples (`ADC_SAMPLES` is 100)
let mut adc_data: [u32; ADC_SAMPLES]
  = [0; ADC_SAMPLES];  //  Init array to 100 zeroes

//  Copy the read ADC Samples to the array
unsafe {                    //  Unsafe because we are copying raw memory
  core::ptr::copy(          //  Copy the memory...
    (*ctx).channel_data,    //  From Source (ADC DMA data)
    adc_data.as_mut_ptr(),  //  To Destination (mutable pointer to adc_data)
    adc_data.len()          //  Number of Items (each item is uint32 or 4 bytes)
  );    
}

Note the differences...

  1. For Rust the Source Pointer is the first parameter, followed by the Destination Pointer

    (This is flipped from memcpy in C)

  2. For Rust the third parameter is the Number of Items to be copied. (100 items)

    (For memcpy the third parameter specifies the number of bytes to copy, i.e. 400 bytes)

Copy ADC data in Rust

Appendix: Generating the Rust Wrapper for BL602 IoT SDK

How was the BL602 Rust Wrapper generated for publishing on crates.io?

Two tools were used to generate the Rust Wrapper for BL602 IoT SDK...

  1. bindgen: Command-line tool that generates the Rust Bindings for a C API.

    (As specified by C Header Files)

  2. safe_wrap: A Rust Procedural Macro we wrote to transform the BL602 C Types to safer Rust-Friendly Types.

    (Including the "expect" checking for return values)

Here are the steps for generating the Rust Wrapper...

##  Install bindgen and clang: https://rust-lang.github.io/rust-bindgen/requirements.html 
cargo install bindgen
sudo apt install llvm-dev libclang-dev clang

##  Download the source code
git clone --recursive https://github.com/lupyuen/bl602-rust-wrapper
git clone --recursive https://github.com/lupyuen/bl_iot_sdk

##  Generate the Rust Bindings for BL602 IoT SDK
cd bl602-rust-wrapper
scripts/gen-bindings.sh

##  Build the docs and the test project
scripts/build.sh

BL602 Rust Wrapper generated by bindgen and safe_wrap

How it works

This script...

Calls bindgen to read the BL602 IoT SDK Header Files...

//  Function Declarations from BL602 IoT SDK (GPIO HAL)
//  https://github.com/lupyuen/bl_iot_sdk/blob/master/components/hal_drv/bl602_hal/bl_gpio.h
int bl_gpio_enable_output(uint8_t pin, uint8_t pullup, uint8_t pulldown);
int bl_gpio_output_set(uint8_t pin, uint8_t value);

To produce the Rust Bindings for BL602 IoT SDK...

//  Rust Bindings for BL602 GPIO generated by gen-bindings.sh
#[safe_wrap(_)] extern "C" {
    pub fn bl_gpio_enable_output(pin: u8, pullup: u8, pulldown: u8) -> ::cty::c_int;
}

#[safe_wrap(_)] extern "C" {
    pub fn bl_gpio_output_set(pin: u8, value: u8) -> ::cty::c_int;
}

(safe_wrap was inserted by an sed script in gen-bindings.sh)

When the above Rust Bindings are compiled, they invoke the safe_wrap Procedural Macro...

To produce the Rust Wrapper for BL602 IoT SDK...

//  Expanded version of `safe_wrap` macros for the GPIO Rust Bindings
#[doc = "Configure a GPIO Pin for Output Mode. See `bl_gpio_enable_output` in \"Enable GPIO\" <https://lupyuen.github.io/articles/led#enable-gpio>"]
pub fn enable_output(pin: u8, pullup: u8, pulldown: u8)
    -> BlResult<()> {
    "----------Extern Decl----------";
    extern "C" {
        pub fn bl_gpio_enable_output(pin: u8, pullup: u8, pulldown: u8)
        -> ::cty::c_int;
    }
    "----------Validation----------";
    unsafe {
        "----------Call----------";
        let res =
            bl_gpio_enable_output(pin as u8, pullup as u8,
                                    pulldown as u8);
        "----------Result----------";
        match res { 0 => Ok(()), _ => Err(BlError::from(res)), }
    }
}

#[doc = "Set output value of GPIO Pin. See `bl_gpio_output_set` in \"Read and Write GPIO\" <https://lupyuen.github.io/articles/led#read-and-write-gpio>"]
pub fn output_set(pin: u8, value: u8) -> BlResult<()> {
    "----------Extern Decl----------";
    extern "C" {
        pub fn bl_gpio_output_set(pin: u8, value: u8)
        -> ::cty::c_int;
    }
    "----------Validation----------";
    unsafe {
        "----------Call----------";
        let res = bl_gpio_output_set(pin as u8, value as u8);
        "----------Result----------";
        match res { 0 => Ok(()), _ => Err(BlError::from(res)), }
    }
}

(More about doc in the next section)

Note that the safe_wrap macro converts the BL602 return values to a Rust Result Type...

match res { 
    0 => Ok(()), 
    _ => Err(BlError::from(res))
}

Which enables the caller to check for errors with expect.

We build the docs and the test project with this script...

In the previous article we attempted to code the BL602 Rust Wrapper by hand...

BL602 Rust Wrapper Documentation

Inject the docs

How did we create the docs for BL602 Rust Wrapper? (Pic above)

Sadly BL602 IoT SDK doesn't have much documentation... But much of the SDK is already documented in "The RISC-V BL602 Book"!

So we linked each Rust Wrapper Function to the relevant section in "The RISC-V BL602 Book".

We do this through the Rust doc Attribute...

//  Expanded version of `safe_wrap` macros for the GPIO Rust Bindings
#[doc = "Configure a GPIO Pin for Output Mode. See `bl_gpio_enable_output` in \"Enable GPIO\" <https://lupyuen.github.io/articles/led#enable-gpio>"]
pub fn enable_output(pin: u8, pullup: u8, pulldown: u8) { ...

#[doc = "Set output value of GPIO Pin. See `bl_gpio_output_set` in \"Read and Write GPIO\" <https://lupyuen.github.io/articles/led#read-and-write-gpio>"]
pub fn output_set(pin: u8, value: u8) -> BlResult<()> { ...

Documentation links to be injected

How did we inject the doc links into the doc Attribute?

For each Rust Wrapper Function, the links to "The RISC-V BL602 Book" are defined in this Markdown Text File...

| Function              | Description                           | Section             | URL
| --------------------- | ------------------------------------- | ------------------- | ---
| bl_gpio_enable_output | Configure a GPIO Pin for Output Mode. | Enable GPIO         | https://lupyuen.github.io/articles/led#enable-gpio
| bl_gpio_output_set    | Set the output value of a GPIO Pin.   | Read and Write GPIO | https://lupyuen.github.io/articles/led#read-and-write-gpio

When our Rust Firmware is compiled, the safe_wrap macro loads the Markdown File into memory...

Loading the documentation links

(Source)

And injects the doc links into the doc attribute...

Injecting the documentation links

(Source)

And the links to "The RISC-V BL602 Book" will magically appear in the Rust Docs!

Renaming the Rust Functions

Rename the functions

The safe_wrap macro also shortens the names of the Rust Wrapper Functions.

Here's the original function from the BL602 IoT SDK...

bl_gpio_enable_output

And here's the Rust Wrapper function shortened by safe_wrap...

gpio::enable_output

transform_function_name contains the code that renames the functions: safe_wrap.rs

Testing the improvised Light Sensor on PineCone BL602

Testing the improvised Light Sensor on PineCone BL602