Table of Contents

Objectives Group Assignment Individual Assignment

Week 10: Output Devices

Objectives

  • group assignment:
    • measure the power consumption of an output device
  • individual assignment:
    • add an output device to a microcontroller board you've designed
    • and program it to do something

Group Assignment

This week we learnt to calculate power consumed by output devices. Specifically, we tested servos. Click here to access our group assignment

Individual Assignment

Using the Xiao ESP32S3

The pinout for Xiao ESP32S3 are shown here. Additionally, all GPIO pins on Xiao ESP32 can provide PWM pins

While you can use any pin, in the Arduino IDE, the default I2S pins for ESP32S3 are the GPIO 8 (SDA) & GPIO 9 (SCL) pins, which are pins D9, and D10 respectively on the Xiao ESP32-S3

You can use any of the GPIO pins for I2S communication

Using the SSD1306 OLED Display

Source: Datasheet Hub

OLED here stands for Organic Light Emitting Diode, or organic LEDs for short. This means that their key building blocks are organic, ie. made from carbon. Specifically, the device consists of a thin carbon-based semiconductor layer that emits light when electricity is applied accross two electrodes.

Some of the key features of the SSD1306 OLED display include

  • 128x64 resolution
  • I2C Interface (or 3/4 wire Serial Peripheral Interface)
  • Can operate between -40 deg to 85 deg C

The SSD1306 OLED display uses the I2C protocol for communication. I asked ChatGPT to explain more about I2C:

Based on datasheet for SSD1306 OLED Display, the device should be powered with a minimum of 7V ideally. But when I looked at an instructable for reference, the OLED VCC pin was powered by VIN pin on the ESP32Devkit-V1. The VIN pin provides 5V from the USB power supply. So I will be doing the same and provide only 5V to VCC of the OLED Display

Using the MG90S Servo

Essentially, a servo is any DC motor with a feedback mechanism built-in to monitor the position of the shaft.

Source: Sparkfun

A standard hobby servo is usually made of the follwing parts, a DC motor to provide the rotation, a controller circuit which takes in signal information about where to move, and a potentiometer that forms the feedback mechanism that monitors the position.

Some of the key features of a servo include:

  • It requires a Pulse Width Modulation signal to move to angles between 0 and its maximum angle
  • There are two types of servos; a standard hobby servo or closed loop servo will provide a range of 90 or 180 degrees, or coninuous loop servos that lack position control and therefore rotate 360 degrees
  • You can either directly control the servo using your microcontroller, as long as it has pins for power, ground and a signal pin capable of providing PWM pins. Or, the servo can be controlled with dedicated servo drivers

Connecting a servo is very simple, all you need ara a GPIO pin capable of providing PWM, a power pin, and a GND pin.

Source: Instructables

Designing the Board

Schematic

Creating the Schematic

Initially I wanted to add a breakout board to connect to the board I made previously during Week 9: Input Devices, but I had to changed my plans since I realized I provided only 3V to VCC pin on the board.

To be able to use the ESP-SR framework for voice recognition, I will be using the ESP32S3 in the form of the Xiao ESP32S3.

This time I also edited the symbol for the INMP441 using the Symbol Editor

This caused warnings related to broken link to footprint to show up in Electronics Rule Checker, so I reassigned the footprint when relinking the footprint was not enough

For the OLED SSD1306 display, initially I tried adding a custom external symbol & footprint, but it did not work out in time, so I used a simple pin header to make the drill holes. Then I made a mistake in the arrangment of the pins (See Mistakes and Solutions, which was corrected only after milling and assembling the first board

I tried to add (Left) pin headers in the same way that Neil (Right) did to control two servos.

I also added a horizontal 2 pin JST connector called S2B-PH-SM4-TB(LF)(SN). Both the symbol and footprint were downloaded from from SnapEDA

This is my completed schematic.

Schematic

PCB Designing

To make my work faster, this time I imported the constraints from previously developed boards. I made some minor changes in the minimum hole size and later annual width and hole to hole clearance as well.

These are my modified constraints

A tip I learnt today: To select the whole trace, click on part of a trace and press U. Additionally you can Shift + Click on multiple parts of different traces to be deleted, select U to select all of them and press 'Delete' to remove all of them at once.

Most of my Design Rule Checker errors were solved by adjusting annular widths and removing unneccesary holes

This is how my PCB looked like when I made the first board with the error in the holes for the OLED. This is the board I milled and used for programming

PCBDesign

This is how it looks like after the revision. I did not end up milling this version.

Making the PCB

This is how my board looks like assembled completely

Milling

First we export our gerber files, then convert to png files using Gerber2PNG

Then we mill our PCB just like before. However, this time I am using the 1/64" flat end mill to make the traces instead of the 0.2 mm V bit

A good tip would be cross-check that you have tightened both ends of the end mill using the Allen key before starting the milling operation.

Allenkey tightening

Bill of Materials

This is how the BOM sheet provided by the interactive BOM plugin sheet looks like

This is my final BOM sheet before I start soldering

Soldering

For the INMP441 microphone module, I decided to reuse the component from the board I made last week

When I realized I had reversed the order for proper placement of the OLED display, I decided to disconnect the pads from their connetions and add new connections using insulated copper wire.

But as you will observe later on, this did not work out very well, at which point I had to resolder the connections, this time I flipped the position of the OLED so that the copper wires can be added without going one over other, which I suspect caused problems while operating the OLED

Verification

Using the Microscope

Using the Benchtop Power Supply

Debugging Errors relaeted to OLED Display

However, I ran into a problem with the OLED display. On the first day, It ran fine without ny issues with the example code. But on travelling back home, and testing again the following day, it stopped working

  • I used the multimeter to test for any short circuits but I did not find any in the traces connecting the display.
  • I checked for voltage drop between PWR and GND; there were no problems and there was 5V being dropped across the PWR and GND just as designed.

When these two steps did not reveal the problem to me, I referred to the steps of debigging listed by Henk during the recitation on programming and debugging. I have listed a screenshot of their presentation (linked here)below for future reference

I tried reproducing the error I got by loading some code I had made (See here) and added some Serial.print lines to confirm that the code has sucessfuly run

Changing the address from 0xC to 0XD did not change anything. I still got the same message in the Serial Monitor

I asked ChatGPT for suggestions and implemented them but I had no luck

Since I already tested the connections with the multimeter, I moved onto the next step.

Since I did not have a microscope on hand, I used my phone's camera and zoomed in to see if there were any problems with the connections. I saw that a blob of solder had been stuck between the traces, so I removed that, but it did not fix things

This is what we understand so far from my experiments:

  • The code is working; this is proved by the fact that it runs on Wokwi without any issues.
  • The connections are correct and are not short circuited
  • The OLED display device drops 5V when connected to power and programmed just as expected

I am not sure, but I think that if I had a logic analyzer or oscilloscope on hand, maybe I could check to see if SDA and SCL connections are working properly. As of now, my best option seems to be desolder the OLED display and connect to an Arduino I have at home and confirm whether the device itself is malfunctioning.

But much to my surprise and relief, the OLED display worked properly with both 5V and 3.3V input from the Arduino UNO board

This means two things

  • Either, there was some problem with the connections that I was not able to identify
  • Or, the signals provided by the SDA and SCL pins of the XiaoESP32 are faulty.

Either way, I will need to mill a new PCB with modification with the change in the pins

Programming

I was frustrated with how most tutorials on the animating eyes simply showed the code without explaining much about why each step is required, so through my documentation I wanted to try to create a tutorial series from scratch that would take you from knowing nothing to making the animating eyes

The first step is to install both the Adafruit_SSD1306 & Adafruit_GFX libraries from the Arduino IDE library manager

Program 1: Find the I2C address of the OLED Display

The first step to programming on the OLED is using to find the I2C Address of our device, the SSD1306 OLED in our case. For this I am going to adapt the code from this tutorial to do so. To understand more about each line of code, I use the help of ChatGPT

Now that I have some understanding of what the code means, I will rewrite it for my purposes using ESP32S3 microcontroller in Wokwi simulator.

I was hoping to be able to write this code without looking to promote learning the logic behind it, but I had to refer back to the source with every line.

Now to sort some errors I made while coding. Debugging seems to be the most important part of coding, because while trying to fox your code, you tend to learn the best about what each line of code does and the pros and cons of different code logic. Even if you dont truly understand it while writing it, you tend to understand much more while debugging.

This is a video of the working. As you can see the I2C address of the SSD1306 module is 0x3C, which can be confirmed by Wokwi's reference documentation


    #include <Wire.h>

    #define RGB 38
    #define SDA 8
    #define SCL 9
    
    void setup(){
        Serial.begin(115200);
        // while(!Serial){}
        Serial.println("I2C Scanner scanning...");
        byte count=0;
        pinMode(RGB,OUTPUT);
        digitalWrite(RGB,HIGH);
        Wire.begin();

        for (byte i=1; i<120; i++){
        Wire.beginTransmission(i);
        // Serial.println (Wire.endTransmission());
        if (Wire.endTransmission()==0){
            // Serial.println (Wire.endTransmission());
            Serial.print ("Found Address: ");
            Serial.print(i, DEC);
            Serial.print(" (0x");
            Serial.print(i, HEX);
            Serial.println(")");
            count++;
            delay(1);
        }
        }
        Serial.print("Done. ");
        Serial.print("Found ");
        Serial.print(count, DEC);
        Serial.print(" device(s)");
    }
    
    void loop(){}     
                        

Program 2: Add text to your display

I will be following the code examples seen here for reference and to get familiar with programming the OLED display.

To understand the some parts of the code I used the help of ChatGPT and confirmed with one of the examples from Adafruit_SDD1306 library




        #include <Wire.h>
        #include <Adafruit_GFX.h>
        #include <Adafruit_SSD1306.h>
        
        #define Height 128
        #define Width 64
        
        Adafruit_SSD1306 display(Height,Width,&Wire,-1);
        
        void setup(){
            Serial.begin(115200);
        
            if (!display.begin(SSD1306_SWITCHCAPVCC, 0x3C)){
            Serial.println ("I2C initialisation failed");
            for(;;);
            }
            delay(200);
            
            display.clearDisplay();
            display.setTextSize(1);
            display.setTextColor(WHITE);
            display.setCursor(0,0);
            display.println("Hello world!");
            display.display(); 
        }
        
        void loop(){ 
        }                            
                        

Program 3: Display a Facial Expression in the OLED Display

To make life a bit easier, I was going to try to use the FluxGarage RoboEyes library, which is based on the Adafruit GFX Library, so it should work with ESP32 as well. However, it won't be possible to simulate on Wokwi, since only paying customers can add custom libraries that are not yet made available for free in the library manager.

So instead I am going to use this tutorial that uses the U8g2 library, which is supported by Wokwi simulator

Following the tutorial I went to Lopaka.app, selected the U8g2 library, and created a new project

This is a youtube video that shows the basic overview of using the Lopaka app

Whenever I face errors, I use ChatGPT to understand what the error message actually means and what I have to do to solve it






        #include <U8g2lib.h>
        #include <Wire.h>
        
        // #define SCL_PIN 6
        // #define SDA_PIN 5
        // U8G2_SSD1306_128X64_NONAME_F_HW_I2C(rotation, [reset [, clock, data]]) [full framebuffer, size = 1024 bytes]
        
        U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, /*SCL_PIN, SDA_PIN,*/ U8X8_PIN_NONE);
        
        void setup(){
            Serial.begin(9600);
            if (!u8g2.begin()){
            Serial.println("Display not initialized");
            for(;;);
            }
            delay(1000);
            Serial.println("Display initialized");
        }
        
        void eye_open(){
            u8g2.clearBuffer();
            u8g2.setFontMode(1);
            u8g2.setBitmapMode(1);
            u8g2.drawFilledEllipse(36, 23, 12, 18);
            u8g2.drawFilledEllipse(89, 23, 12, 18);
            u8g2.drawFilledEllipse(61, 53, 12, 4);
            u8g2.sendBuffer();
        }
        
        void loop(){
            eye_open();
        }
                        

By this time, I was able to get my code to work the OLED Display using the Arduino UNO, so I switched to working physically with the device

As you can see, the same code worked here as well, but a small adjustment in lowering the height of the eyes might remove the orange colour, leaving only the blue colour

The previous expression was not being overwritten, even when I uploaded new code. The only thing that worked was removing the 5V pin and reconnecting


        #include <U8g2lib.h>
        #include <Wire.>
        
        U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE);
        
        void setup(){
            Serial.begin(9600);
            if (!u8g2.begin()){
            Serial.println("Display not initialized");
            for(;;);
            }
            delay(1000);
            Serial.println("Display initialized");
        }
        
        // Function to Open Eyes
        void eye_open(){
            u8g2.clearBuffer();
            u8g2.setFontMode(1);
            u8g2.setBitmapMode(1);
            u8g2.drawFilledEllipse(83, 33, 8, 13);
            u8g2.drawFilledEllipse(62, 54, 8, 6);
            u8g2.drawFilledEllipse(41, 33, 8, 13);
            u8g2.sendBuffer();
        }
        
        void loop(){
            eye_open();
        }
                                
                        

Program 3: Make a simple face animation

First we copy the eyes into a new screen from Lopaka.app as shown below

Then I made new functions for three expressions, with the mouth at slightly different orientations. I added a delay time in between them that I randomized using random() function. I then used the same function to randomize the vertical height of the three frames, creating a more lifelike animation.

Unfortunately, the limitation with lopaka app is that you can only use three screens in the free plan, so you will need to take your code into the editor, then delete the screen and create a new one.

    
        #include <U8g2lib.h>
        #include <Wire.h>
        
        long dt;
        long vert;
        
        U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE);
        
        void setup(){
            Serial.begin(115200);
            if (!u8g2.begin()){
            Serial.println("Display not initialized");
            for(;;);
            }
            delay(1000);
            Serial.println("Display initialized");
        }
        
        void mouth_open(){
            u8g2.clearBuffer();
            u8g2.setFontMode(1);
            u8g2.setBitmapMode(1);
            u8g2.drawFilledEllipse(83, 33+vert, 8, 13);
            u8g2.drawFilledEllipse(62, 54+vert, 8, 6);
            u8g2.drawFilledEllipse(41, 33+vert, 8, 13);
            u8g2.sendBuffer();
        }
        
        void mouth_half_open (){
            u8g2.clearBuffer();
            u8g2.setFontMode(1);
            u8g2.setBitmapMode(1);
            u8g2.drawFilledEllipse(83, 33+vert, 8, 13);
            u8g2.drawFilledEllipse(41, 33+vert, 8, 13);
            u8g2.drawFilledEllipse(62, 57+vert, 8, 3);
            u8g2.sendBuffer();
        } 
        
        void mouth_closed (){
            u8g2.clearBuffer();
            u8g2.setFontMode(1);
            u8g2.setBitmapMode(1);
            u8g2.drawFilledEllipse(83, 33+vert, 8, 13);
            u8g2.drawFilledEllipse(62, 59+vert, 8, 1);
            u8g2.drawFilledEllipse(41, 33+vert, 8, 13);
            u8g2.sendBuffer();
        }
        
        void loop(){
            vert = random(0,3);
            dt = random(0,150);
            
            mouth_closed();
            delay(dt);
            // Serial.println(dt);
        
            vert = random(0,3);
            dt = random(0,200);
            mouth_half_open();
            delay(dt);
            // Serial.println(dt);
        
            vert = random(0,3);
            dt = random(0,200);
            mouth_open();
            delay(dt);
            // Serial.println(dt);
        
            vert = random(0,3);
            dt = random(0,200);
            mouth_half_open();
            delay(dt);
            // Serial.println(dt);
        
            vert = random(0,3);
            dt = random(0,200);
            mouth_closed();
            delay(dt);
            // Serial.println(dt);
        }
            
                        

Later once I got the board to work; I got the code to work on it as well

Program 5: Adding voice control to my display

Since my project is supposed to be voice activated, I wanted to add some kind of voice input to my OLED display

First I played around with the code to make it more efficient by adding variable that change the movement of the mouth instead of calling three different functions for the same thing. Right now the mouth moves too fast because the timing is set by the same delay timing as the eyes, but Its not an issue since I am going to be controlling the mouth using the mic.

    
        #include <U8g2lib.h>
        #include <Wire.h>
        
        long dt;
        long vert;
        long mouthC; //Center of ellipse of mouth
        long mouthV; //Radius of ellipse of mouth in Y axis
        
        U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE);
        
        void setup(){
            Serial.begin(115200);
            if (!u8g2.begin()){
            Serial.println("Display not initialized");
            for(;;);
            }
            delay(1000);
            Serial.println("Display initialized");
        }
        
        void mouth_open(){
            u8g2.clearBuffer();
            u8g2.setFontMode(1);
            u8g2.setBitmapMode(1);
            u8g2.drawFilledEllipse(83, 30+vert, 8, 13);
            u8g2.drawFilledEllipse(62, mouthC, 8, mouthV);
            u8g2.drawFilledEllipse(41, 30+vert, 8, 13);
            u8g2.sendBuffer();
        }
        
        void loop(){
            vert = random(0,3);
            dt = random(0,150);
            mouthC = random (49,54);
            mouthV = random (2,6);
            
            mouth_open();
            dt = random(0,150);
        }
                        

The cut accross the eyes only appears on video (I assume because of the frame rate differences between the phone camera at 30 FPS and the OLED). It does not appear in rea life.

I used this tutorial to help figure out how to operate the INMP441 mic. Sice the code is outdated, I used their github documentation and Revishankar's (FA25) documentation to write my code


    #include <driver/i2s.h>
    #define I2S_WS 8
    #define I2S_SCK 9
    #define I2S_SD 7
    #define I2S_PORT I2S_NUM_0 //choosing I2S0 controller 
    #define bufferLen 64

    int16_t sBuffer[bufferLen]; //int16_t is data type for 2 byte signed integers

    void i2s_install(){

        const i2s_config_t i2s_config = {
            .mode = i2s_mode_t(I2S_MODE_MASTER | I2S_MODE_RX),
            .sample_rate = 44100,
            .bits_per_sample = i2s_bits_per_sample_t(16),
            .channel_format = I2S_CHANNEL_FMT_ONLY_LEFT,
            //Configures I2S to use only the left channel, since we have grounded the L/R pin.
            //To use ..ONLY_RIGHT,  leave L/R pin floating
            .communication_format = i2s_comm_format_t(I2S_COMM_FORMAT_STAND_I2S),
            .intr_alloc_flags = 0, // default interrupt priority
            .dma_buf_count = 8,
            .dma_buf_len = bufferLen,
            .use_apll = false
        };

    i2s_driver_install(I2S_PORT, &i2s_config, 0, NULL);
    }

        void i2s_setpin(){
        const i2s_pin_config_t pin_config = {
            .bck_io_num = I2S_SCK,
            .ws_io_num = I2S_WS,
            .data_out_num = -1,
            .data_in_num = I2S_SD
        };

    i2s_set_pin(I2S_PORT, &pin_config);
    }

    void setup() {
        pinMode(LED_BUILTIN,OUTPUT);

        Serial.begin(115200);
        Serial.println("Setup I2S ...");

        delay(1000);
        i2s_install();
        i2s_setpin();
        i2s_start(I2S_PORT);
        Serial.println("I2S Setup successful ...");
        delay(50);
    
    }

    void loop() {

        size_t bytesIn=0;
        esp_err_t result = i2s_read(I2S_PORT, &sBuffer, bufferLen, &bytesIn, portMAX_DELAY);
        if (result == ESP_OK)
        {
            int samples_read = bytesIn/bufferLen;
            int16_t sum=0;
            float mean = 0;

            if (samples_read>0)
            {
                for (int i=0; i<samples_read; i++)
                    {
                        mean += (sBuffer[i]);
                    }
                mean/=samples_read; //shorthand for mean=mean/samples_read
                Serial.println(mean);
            }

            if (mean>200){
                digitalWrite(LED_BUILTIN, LOW);
                delay(500);
                }

            else{
                digitalWrite(LED_BUILTIN, HIGH);
                }

            // Serial.print("Amplitude: ");
            Serial.println(mean);
            delay(100);
        }
    }

                        

Current Progress: Currently I am able to manipulate the opening of the mouth using my voice, but I need to make the code much better.

    #include <U8g2lib.h>
    #include <Wire.h>
    #include <driver/i2s.h>
    #define I2S_WS 8
    #define I2S_SCK 9
    #define I2S_SD 7
    #define I2S_PORT I2S_NUM_0 //choosing I2S0 controller 
    #define bufferLen 64
    
    int16_t sBuffer[bufferLen]; //int16_t is data type for 2 byte signed integers
    long dt;
    long vert;
    long mouthC; //Center of ellipse of mouth
    long mouthV; //Radius of ellipse of mouth in Y axis
    
    U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE);
    
    void i2s_install(){
    
        const i2s_config_t i2s_config = {
        .mode = i2s_mode_t(I2S_MODE_MASTER | I2S_MODE_RX),
        .sample_rate = 44100,
        .bits_per_sample = i2s_bits_per_sample_t(16),
        //Configures I2S to use only the left channel, since we have grounded the L/R pin. To use ..ONLY_RIGHT, leave L/R pin floating
        .channel_format = I2S_CHANNEL_FMT_ONLY_LEFT, 
        .communication_format = i2s_comm_format_t(I2S_COMM_FORMAT_STAND_I2S),
        .intr_alloc_flags = 0, // default interrupt priority
        .dma_buf_count = 8,
        .dma_buf_len = bufferLen,
        .use_apll = false
        };
    
    i2s_driver_install(I2S_PORT, &i2s_config, 0, NULL);
    }
    
    void i2s_setpin(){
        const i2s_pin_config_t pin_config = {
        .bck_io_num = I2S_SCK,
        .ws_io_num = I2S_WS,
        .data_out_num = -1,
        .data_in_num = I2S_SD
        };
    
    i2s_set_pin(I2S_PORT, &pin_config);
    }
    
    
    void setup(){
        Serial.begin(115200);
        pinMode(LED_BUILTIN,OUTPUT);
    
        Serial.println("Setup I2S ...");
    
        delay(1000);
        i2s_install();
        i2s_setpin();
        i2s_start(I2S_PORT);
        Serial.println("I2S Setup successful ...");
        delay(50);
    
        if (!u8g2.begin()){
        Serial.println("Display not initialized");
        for(;;);
        }
        else {
            Serial.println("Display initialized");
        }
        delay(1000);
    }
    
    void mouth_open(){
        u8g2.clearBuffer();
        u8g2.setFontMode(1);
        u8g2.setBitmapMode(1);
        u8g2.drawFilledEllipse(83, vert, 8, 13);
        u8g2.drawFilledEllipse(62, mouthC, 8, mouthV);
        u8g2.drawFilledEllipse(41, vert, 8, 13);
        u8g2.sendBuffer();
    }
    
    void loop(){
        vert = random(30,33);
        dt = random(0,150);
        mouthC = random (49,54);
        mouthV = random (2,6);
    
        size_t bytesIn=0;
        esp_err_t result = i2s_read(I2S_PORT, &sBuffer, bufferLen, &bytesIn, portMAX_DELAY);
        if (result == ESP_OK)
        {
        int samples_read = bytesIn/bufferLen;
        int16_t sum=0;
        float mean = 0;
    
        if (samples_read>0)
        {
            for (int i=0; i200){
            mouthV = map (mean, 200, 600, 2, 6);
            mouth_open();
            dt = random(0,150);
        }
    
        else{
            digitalWrite(LED_BUILTIN, HIGH);
            Serial.println ("Voice not detected");
            // delay(1000);
        }
    
        Serial.println(mean);
        delay(50);
        }
    }
                    

Program 6: Controlling a Servo Motor using the Xiao ESP32S3

Since the standard Arduino Servo library is not compatible, I have to use a different library called ESP32Servo. Using the example Sweep code did work, but it did not rotate 180 degrees like expected, instead the servo rotated to around 90 degrees before sweeping back. So we need to adjust the minimum and maximum pulse widths. According to the datasheet, the MG90S has a duty cycle of 1000 µs and minimum and maximum pulse widths of 1000-2000 µs. I found another reference from the internet that said that the parameters should be 500 µs & 2500 µs respectively

I added some code to be used to calibrate the '0° position' of the servo that will be commented later. By changing 'pos' value, I can confirm that the servo is rotating till 180°

To understand the allocate.Timer code line, I used the help of ChatGPT and ESP32PWM Class reference





        #include <ESP32Servo.h>
        Servo myServo;
        
        #define sPin 43
        #define PULSEMIN 500
        #define PULSEMAX 2500
        int pos = 180;
        
        void setup(){
            Serial.begin(9600);
            pinMode (sPin, OUTPUT);
            ESP32PWM::allocateTimer(0);
            myServo.attach (sPin, PULSEMIN, PULSEMAX);
        }
        
        void loop (){
            // //To caibrate
            // myServo.write(pos);
            // Serial.print("Servo position: ")
            // Serial.println(pos);
        
            // To sweep
            for (pos=0; pos<=180; pos+=1){
            myServo.write(pos);
            Serial.print("Servo position: ");
            Serial.println(pos);
            delay(15);
            
            }
            delay(1000); //Stops for a second after reaching 180 degree
        
            for (pos=180; pos>=0; pos-=1) {
            myServo.write(pos);
            Serial.print("Servo position: ");
            Serial.println(pos);
            delay(15); 
            }
            delay(1000); //Stops for a second after reaching 0 degree   
        }
            
                        

Conclusion

This week I learnt to:

  • Measure power consumption of a device using the bench power supply
  • Program an I2C output device; the SSD1306 OLED display
  • Program a servo

Mistakes & Solutions

  1. Adding SSD1306 Symbol and Footprint: I tried using this library I found for SSD1306 128x64. To add the symbol, I added the SSD1306-128x64_OLED.lib file, and to add the footprint, I added the SSD1306.pretty folder.

    But I ended up replacing this with a simple pin header, when the link between the symbol and footprint got broken and I was not able to relink or reassign it

  2. Editing Symbols and Footprints: If you want to assign names to pins, change the pin name and the pin value in the symbol editor, then in the PCB editor, go to the footprint editor and add the same numbers you assigned previously in the schematic. Once you update the PCB, you will see that the airwires connections will be visible.
  3. Annular Width DRC Error: Even though I set the constraints and the netclass for hole paramters, I still had to manually adjust the annular widths of the drill holes. Currently I do not know the solution to this problem execpt manually adjusting pad width in the X and Y direction in Pad properties and also adjusting annular width constraints if required.

  4. Crosscheck if both screws are tightened once before milling
  5. Crosscheck if you have correct PNG files once before milling
  6. Check the orientation of the OLED display designing the PCB: I did not think about the position of the display, which forced me to do one of three things:

    1. Attach the OLED display as is on top, meaning it would cover the ESP32 MCU: But this was not an option since it would block access to the pins for the servos.
    2. Attach the OLED display on the back of the PCB: This would make it a bit more tricker to handle the PCB while programming, so I wanted to avoid this option
    3. Short the connections using a blade and copper wire: This is what I ended up doing, but the OLED display only worked once like this. By the second day, it stopped responding. I suspect that the wires may have shorted with each other or the solder may have come loose on some of the connections

    In hindsight, I should have gone forward with option 2 to avoid problems related to using copper wires to make new connections

  7. Debugging errors related to OLED display: When you come accross an issue of a device not working, following a step by step approach to find the issue definitely helps. Following this checklist provide by Henk during the recitation 'Debugging and Programming & Debugging' and using ChatGPT to debug your code helped me eliminate many potential causes of errors.

    I used the Serial Monitor, multimeter, and my phone camera during the process of debugging. When I was able to successfully able to program the display when I connected it to an Arduino UNO board I had on hand, I was able to isolate 2 probable causes of my problem.

    • Either, there was some problem with the connections that I was not able to identify
    • Or, the signals provided by the SDA and SCL pins of the XiaoESP32 are faulty.

  8. Inverted output for LED_BUILTIN: According to Seeed Studio Wiki : The LED will only turn off when the user LED pin on the XIAO ESP32S3 is set to a high level, and it will only turn on when the pin is set to a low level.

    (ie. digitalWrite(LED_BUILTIN, HIGH) will turn the LED OFF, and digitalWrite(LED_BUILTIN, LOW) will turn the LED ON

  9. American vs British spellings for commands: Remember to check the spellings of the commands. Depending on the creator of the library you want to use, they may use British or American standards while naming their commands. As you can see, the Adafruit_SSD1306 library uses American English, so the correct way to spell the command to change the colour is shown below
  10. Tip: Use for(;;) code to make an infinite loop: Much easier way of adding an infinite loop in your code using just one line of code.

Design Files

Click here to access the design files