-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
7c9590e
commit 347fea4
Showing
16 changed files
with
506 additions
and
15 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,131 @@ | ||
## Connecting HC-SR04 ultrasonic sensor with ESP32 | ||
|
||
|
||
### Why We Need a Voltage Divider? | ||
|
||
You can skip this, if you have HC-SR04+ which accepts 3.3V power supply also. | ||
|
||
Before diving into the connection, You should familiarize yourself with the concept of a voltage divider; You can refer to [this chapter](../core-concepts/voltage-divider.md). As we mentioned earlier, we need to power the module using a 5V power supply, which can be supplied through the Vin pin of the ESP32. The module sends back the signal through the Echo pin. However, the ESP32 is only around 3.6V tolerant on its GPIO pins, so we need to reduce the voltage using a voltage divider between the Echo pin and the ESP32's GPIO pin. | ||
|
||
To make this work, you'll need two resistors with different values, ensuring the output voltage is approximately 3.3V. For example, you can use a 1kΩ resistor as R1 and a 2kΩ resistor as R2, which will bring the voltage down to around 3.3V. | ||
|
||
<img style="display: block; margin: auto;" alt="ultrasonic" src="./images/voltage-divider-hc-sr04-3_3_v.png"/> | ||
|
||
If you want to experiment with different resistor values, you can use the [Falstd website](https://www.falstad.com/circuit/) with this [voltage-divider circuit text file](./voltage-divider-hc-sr04.txt). It allows you to modify the values of R1 and R2 to see what voltage each combination will output. This way, you can create the voltage divider with the resistors you have on hand. | ||
|
||
## Circuit for HC-SR04 | ||
|
||
<table> | ||
<thead> | ||
<tr> | ||
<th>ESP32 Pin</th> | ||
<th style="height: 4px; width: 250px; margin: 0 auto;">Wire</th> | ||
<th>HC-SR04 Pin</th> | ||
</tr> | ||
</thead> | ||
<tbody> | ||
<tr> | ||
<td>5V</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire red" style="height: 4px; width: 200px; margin: 0 auto;"> | ||
<div class="male-left"></div> | ||
<div class="male-right"></div> | ||
</div> | ||
</td> | ||
<td>VCC</td> | ||
</tr> | ||
<tr> | ||
<td>GPIO 5</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire green" style="height: 4px; width: 200px; margin: 0 auto;"> | ||
<div class="male-left"></div> | ||
<div class="male-right"></div> | ||
</div> | ||
</td> | ||
<td>Trig</td> | ||
</tr> | ||
<tr> | ||
<td>GPIO 18</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire yellow" style="height: 4px; width: 200px; margin: 0 auto;"> | ||
<div class="male-left"></div> | ||
<div class="male-right"></div> | ||
</div> | ||
</td> | ||
<td>Echo (via voltage divider)</td> | ||
</tr> | ||
<tr> | ||
<td>GND</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire black" style="height: 4px; width: 200px; margin: 0 auto;"> | ||
<div class="male-left"></div> | ||
<div class="male-right"></div> | ||
</div> | ||
</td> | ||
<td>GND</td> | ||
</tr> | ||
</tbody> | ||
</table> | ||
|
||
|
||
- **VCC**: Connect the VCC pin on the HC-SR04 to the Vin pin on the ESP32. If you are using HC-SR04+, use 3.3V pin on the ESP32. | ||
- **Trig**: Connect to GPIO 5 on the ESP32. | ||
- **Echo**: Connect the Echo pin on the HC-SR04 to GPIO 18 through a voltage divider (1kΩ resistor between Echo and GPIO 18, and 2kΩ resistor between GPIO 18 and GND; means the GPIO 18 basically goes in the middle). I believe this will be easier to understand with the circuit diagram. | ||
- **GND**: Connect the GND pin on the HC-SR04 to the GND pin on the ESP32. | ||
|
||
## Circuit for LED | ||
|
||
You have to connect the anode (long leg) of the LED to GPIO 33, as in the [External LED setup](../led/external-led.md); through the resistor (eg: 330 Ohm resistor) to avoid damaging the LED. And the cathode of the LED(short leg) to Ground. | ||
|
||
<table> | ||
<thead> | ||
<tr> | ||
<th>ESP32 Pin</th> | ||
<th style="width: 250px; margin: 0 auto;">Wire</th> | ||
<th>Component</th> | ||
</tr> | ||
</thead> | ||
<tbody> | ||
<tr> | ||
<td>GPIO 33</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire orange" style="width: 200px; margin: 0 auto;"> | ||
<div class="female-left"></div> | ||
<div class="female-right"></div> | ||
</div> | ||
</td> | ||
<td>Resistor</td> | ||
</tr> | ||
<tr> | ||
<td>Resistor</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire orange" style="width: 200px; margin: 0 auto;"> | ||
<div class="female-left"></div> | ||
<div class="female-right"></div> | ||
</div> | ||
</td> | ||
<td>Anode (long leg) of LED</td> | ||
</tr> | ||
<tr> | ||
<td>GND</td> | ||
<td style="text-align: center; vertical-align: middle; padding: 0;"> | ||
<div class="wire black" style="width: 200px; margin: 0 auto;"> | ||
<div class="female-left"></div> | ||
<div class="female-right"></div> | ||
</div> | ||
</td> | ||
<td>Cathode (short leg) of LED</td> | ||
</tr> | ||
</tbody> | ||
</table> | ||
|
||
## Circuit Diagram | ||
|
||
I have provided circuit diagrams both with and without a breadboard. To be honest, the breadboard version was a bit confusing when I drew it. If you still find it unclear, please create an issue in the GitHub repository and describe the confusing parts. I'll do my best to improve it. | ||
|
||
**Diagram without breadboard**: | ||
|
||
<img style="display: block; margin: auto;" alt="connecting ESP32 with HC-SR04 Ultrasonic Sensor circuit" src="./images/ESP32-HC-SR04-circuit-without-breadboard.png"/> | ||
|
||
**Diagram with breadboard**: | ||
<img style="display: block; margin: auto;" alt="connecting ESP32 with HC-SR04 Ultrasonic Sensor circuit" src="./images/ESP32-HC-SR04-circuit.png"/> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,225 @@ | ||
## Writing Rust Code Use HC-SR04 Ultrasonic Sensor with ESP32 | ||
|
||
We'll start by generating the project using the template, then modify the code to fit the current project's requirements. | ||
|
||
|
||
### Generate project using esp-generate | ||
|
||
You have done this step already in the quick start section. | ||
|
||
To create the project, use the `esp-generate` command. Run the following: | ||
|
||
```sh | ||
esp-generate --chip esp32 ultrasonic | ||
``` | ||
|
||
This will open a screen asking you to select options. For now, we dont need to select any options. Just save it by pressing "s" in the keyboard. | ||
|
||
## Setup the LED Pin and configure PWM | ||
You should understand this code by now. If not, please complete the Fading [LED section](../led/index.md) first. | ||
|
||
Quick recap: Here, we're configuring the PWM for the LED, which allows us to control the brightness by adjusting the duty cycle. | ||
|
||
```rust | ||
let led = peripherals.GPIO33; | ||
let ledc = Ledc::new(peripherals.LEDC); | ||
let mut hstimer0 = ledc.timer::<HighSpeed>(timer::Number::Timer0); | ||
hstimer0 | ||
.configure(timer::config::Config { | ||
duty: timer::config::Duty::Duty5Bit, | ||
clock_source: timer::HSClockSource::APBClk, | ||
frequency: 24.kHz(), | ||
}) | ||
.unwrap(); | ||
|
||
let mut channel0 = ledc.channel(channel::Number::Channel0, led); | ||
channel0 | ||
.configure(channel::config::Config { | ||
timer: &hstimer0, | ||
duty_pct: 10, | ||
pin_config: channel::config::PinConfig::PushPull, | ||
}) | ||
.unwrap(); | ||
``` | ||
|
||
## Setup the Trigger Pin | ||
We will configure GPIO 5 as an output pin with its initial state set to LOW. If you're wondering why it's an output, it's because we are sending a signal from the ESP32 to the ultrasonic module. This pin is connected to the Trig pin of the ultrasonic module. | ||
|
||
```rust | ||
let mut trig = Output::new(peripherals.GPIO5, Level::Low); | ||
``` | ||
|
||
## Setup the Echo Pin | ||
We will configure GPIO 18 as an input pin since the ultrasonic module sends the signal back to the ESP32. The initial state of this pin will be set to Pull Down to ensure it starts in the low state. | ||
|
||
```rust | ||
let echo = Input::new(peripherals.GPIO18, Pull::Down); | ||
``` | ||
|
||
## 🦇 Light it Up | ||
|
||
### Step 1: Send the Trigger Pulse | ||
We will set the trig pin to LOW so that we start fresh. We will set the trig pin to HIGH for 10 microseconds, then turn it back to LOW. This will trigger the module to send ultrasonic waves. | ||
|
||
```rust | ||
// Ensure the Trigger pin is low before starting | ||
trig.set_low(); | ||
delay.delay_micros(2); | ||
|
||
// Send a 10-microseconds high pulse | ||
trig.set_high(); | ||
delay.delay_micros(10); | ||
trig.set_low(); | ||
``` | ||
|
||
### Step 2: Measure the pulse width | ||
|
||
Next, we will use two loops. The first loop will run as long as the echo pin state is LOW. Once it goes HIGH, we will record the current time in a variable. Then, we start the second loop, which will continue as long as the echo pin remains HIGH. When it returns to LOW, we will record the current time in another variable. The difference between these two times gives us the pulse width. | ||
|
||
```rust | ||
// Measure the duration the signal remains high | ||
while echo.is_low() {} | ||
let time1 = rtc.current_time(); | ||
while echo.is_high() {} | ||
let time2 = rtc.current_time(); | ||
let pulse_width = match (time2 - time1).num_microseconds() { | ||
Some(pw) => pw as f64, | ||
None => continue, | ||
}; | ||
``` | ||
|
||
### Step 3: Calculate the Distance | ||
|
||
To calculate the distance, we need to use the pulse width. The pulse width tells us how long it took for the ultrasonic waves to travel to an obstacle and return. Since the pulse represents the round-trip time, we divide it by 2 to account for the journey to the obstacle and back. | ||
|
||
The speed of sound in air is approximately 0.0343 cm per microsecond. By multiplying the time (in microseconds) by this value and dividing by 2, we obtain the distance to the obstacle in centimeters. | ||
|
||
```rust | ||
let distance = (pulse_width * 0.0343) / 2.0; | ||
``` | ||
|
||
### Step 4: PWM Duty cycle for LED | ||
Finally, we adjust the LED brightness based on the measured distance. | ||
|
||
The duty cycle percentage is calculated using our own logic, you can modify it to suit your needs. When the object is closer than 30 cm, the LED brightness will increase. The closer the object is to the ultrasonic module, the higher the calculated ratio will be, which in turn adjusts the duty cycle. This results in the LED brightness gradually increasing as the object approaches the sensor. | ||
|
||
```rust | ||
let duty_pct: u8 = if distance < 30.0 { | ||
let ratio = (30.0 - distance) / 30.0; | ||
let p = (ratio * 100.0) as u8; | ||
p.min(100) | ||
} else { | ||
0 | ||
}; | ||
|
||
if let Err(e) = channel0.set_duty(duty_pct) { | ||
esp_println::println!("Failed to set duty cycle: {:?}", e); | ||
} | ||
``` | ||
|
||
### Full code | ||
|
||
```rust | ||
#![no_std] | ||
#![no_main] | ||
|
||
use esp_backtrace as _; | ||
use esp_hal::{ | ||
delay::Delay, | ||
gpio::{Input, Level, Output, Pull}, | ||
ledc::{ | ||
channel::{self, ChannelIFace}, | ||
timer::{self, TimerIFace}, | ||
HighSpeed, Ledc, | ||
}, | ||
prelude::*, | ||
rtc_cntl::Rtc, | ||
}; | ||
|
||
#[entry] | ||
fn main() -> ! { | ||
let peripherals = esp_hal::init({ | ||
let mut config = esp_hal::Config::default(); | ||
config.cpu_clock = CpuClock::max(); | ||
config | ||
}); | ||
|
||
// let led = peripherals.GPIO2; // uses onboard LED | ||
let led = peripherals.GPIO33; | ||
let ledc = Ledc::new(peripherals.LEDC); | ||
let mut hstimer0 = ledc.timer::<HighSpeed>(timer::Number::Timer0); | ||
hstimer0 | ||
.configure(timer::config::Config { | ||
duty: timer::config::Duty::Duty5Bit, | ||
clock_source: timer::HSClockSource::APBClk, | ||
frequency: 24.kHz(), | ||
}) | ||
.unwrap(); | ||
|
||
let mut channel0 = ledc.channel(channel::Number::Channel0, led); | ||
channel0 | ||
.configure(channel::config::Config { | ||
timer: &hstimer0, | ||
duty_pct: 10, | ||
pin_config: channel::config::PinConfig::PushPull, | ||
}) | ||
.unwrap(); | ||
|
||
// For HC-SR04 Ultrasonic | ||
let mut trig = Output::new(peripherals.GPIO5, Level::Low); | ||
let echo = Input::new(peripherals.GPIO18, Pull::Down); | ||
|
||
let delay = Delay::new(); | ||
let rtc = Rtc::new(peripherals.LPWR); | ||
|
||
loop { | ||
delay.delay_millis(5); | ||
|
||
// Trigger ultrasonic waves | ||
trig.set_low(); | ||
delay.delay_micros(2); | ||
trig.set_high(); | ||
delay.delay_micros(10); | ||
trig.set_low(); | ||
|
||
// Measure the duration the signal remains high | ||
while echo.is_low() {} | ||
let time1 = rtc.current_time(); | ||
while echo.is_high() {} | ||
let time2 = rtc.current_time(); | ||
let pulse_width = match (time2 - time1).num_microseconds() { | ||
Some(pw) => pw as f64, | ||
None => continue, | ||
}; | ||
|
||
// Derive distance from the pulse width | ||
let distance = (pulse_width * 0.0343) / 2.0; | ||
// esp_println::println!("Pulse Width: {}", pulse_width); | ||
// esp_println::println!("Distance: {}", distance); | ||
|
||
// Our own logic to calculate duty cycle percentage for the distance | ||
let duty_pct: u8 = if distance < 30.0 { | ||
let ratio = (30.0 - distance) / 30.0; | ||
let p = (ratio * 100.0) as u8; | ||
p.min(100) | ||
} else { | ||
0 | ||
}; | ||
|
||
if let Err(e) = channel0.set_duty(duty_pct) { | ||
esp_println::println!("Failed to set duty cycle: {:?}", e); | ||
} | ||
|
||
delay.delay_millis(60); | ||
} | ||
} | ||
|
||
``` | ||
|
||
## Clone the existing project | ||
You can clone (or refer) project I created and navigate to the `ultrasonic` folder. | ||
|
||
```sh | ||
git clone https://github.com/ImplFerris/esp32-projects/ultrasonic | ||
cd esp32-projects/ultrasonic | ||
``` |
Oops, something went wrong.