-
Notifications
You must be signed in to change notification settings - Fork 7.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
RMT Group Clock too fast (IDFGH-13921) #14760
Comments
As far as I can tell, if the filter_reg_value was stored in a 16-bit register, or if I had access to a slower group clock, I could do what I am attempting. Doesn't seem like 420uS and 1.4mS are unrealistic values and the RMT should be able to support it. |
You should not have issues or need to manually poke into register values. If you use the newer RMT APIs you can have 1uSec resolution with the following TX config:
RX should be similar. |
@atanisoft Thank you for the suggestion. I made the switch from Arduino v2 to ESP-IDF v5 before Arduino v3 was to be released. Figured with API breaking changes, it was the best time to make the move. Here is the API page I've been referencing: https://docs.espressif.com/projects/esp-idf/en/v5.3.1/esp32s3/api-reference/peripherals/rmt.html I believe I have been using the correct values in the calculation, but if it isn't working, must not be correct. I am using the clk_src, 'RMT_CLK_SRC_APB' (80MHz) and my minimum signal pulse is 420,000nS. Using the clk_src and signal_range_min_ns, we can calculate the filter_reg_value. Here is the underlying code on line 393 of https://github.com/espressif/esp-idf/blob/v5.3.1/components/esp_driver_rmt/src/rmt_rx.c
So, plugging in the values:
Which seems legit, since 33,600 will fit in a uint32_t. But looking a bit further at line 395:
So, filter_reg_value needs to be less than or equal to RMT_LL_MAX_FILTER_VALUE, which is defined as
I've spent hours digging and doing trial and error. I'm just waiting for an knowledgeable admin to tell me it's not possible. Almost all of our product use this timing protocol, so it'll be a huge hit. Thank you in advance! |
You seem to be mixing the low level and higher level APIs and going in circles. You can use strictly the higher level APIs as seen in this example to handle RX. For filtering you would use something like this except as:
This will filter out any pulses that are outside the 420-1400uSec range automatically. You could then process the received samples as seen in the example. You can ignore the TX / encoder pieces from the example since you are focusing on RX only. |
I copied the IR NEC transceiver over to my workspace, set-target to esp32s3, then menuconfig to set my flash size, then compiled and flashed. Here is the original unchanged rmt_receive_config_t declaration and initialization:
And here is the snippet from the terminal:
Then I changed
to
And here is the terminal:
I didn't know about the 8-bit limitation until I saw GitHub issue #11262 That information really should be in the API reference. |
In that case you might just set the minimum to zero and handle the filtering in your RX parser (the NEC example validates the values are in a certain range). |
@atanisoft Very good idea! Thanks! You actually reminded my of something I thought of on my drive home yesterday. I'll only see glitches when nothing is transmitted. But during the actual transmission, they are pretty stable. And, for anyone following along, by setting it to zero, means division by 256, as per:
I'll keep this open and update it when I figure it out or quit. |
I've decided to throw in the towel. The ESP32 is amazing until it doesn't work. And I just keep find more and more deficiencies. I'm trying to make the push from programming PIC chips in assembly when the senior engineer retires. But I can't sell this if I am constantly doing "work arounds" to get done what it should be easy and quick, within a acceptable timeframe. And I don't believe you, @atanisoft , work for Espressif, but thank you for helping me. I never know if the issues I post are written horribly or if the question is so good, no one knows, not even Espressif. @Kainarx Last shot since I see it is assigned to you. Any suggestions? |
Hi @dpnebert Thanks for reporting the issue. The root cause is that the driver didn't provide a way to allow the user to set a clock divider for the group's clock. See: https://github.com/espressif/esp-idf/blob/master/components/esp_driver_rmt/src/rmt_common.c#L215 Here is a quick fix to set the group clock resolution to 500KHz --- a/components/esp_driver_rmt/src/rmt_common.c
+++ b/components/esp_driver_rmt/src/rmt_common.c
@@ -209,12 +209,12 @@ esp_err_t rmt_select_periph_clock(rmt_channel_handle_t chan, rmt_clock_source_t
#endif // CONFIG_PM_ENABLE
esp_clk_tree_enable_src((soc_module_clk_t)clk_src, true);
- // no division for group clock source, to achieve highest resolution
+ // Divide group clock source by 80 to achieve a smaller clock resolution
RMT_CLOCK_SRC_ATOMIC() {
- rmt_ll_set_group_clock_src(group->hal.regs, channel_id, clk_src, 1, 1, 0);
+ rmt_ll_set_group_clock_src(group->hal.regs, channel_id, clk_src, 80, 1, 0);
rmt_ll_enable_group_clock(group->hal.regs, true);
}
- group->resolution_hz = periph_src_clk_hz;
+ group->resolution_hz = periph_src_clk_hz / 80;
ESP_LOGD(TAG, "group clock resolution:%"PRIu32, group->resolution_hz);
return ret;
} Then when you select the XTAL as the clock source, the The channel resolution can be set to 100KHz (can't be bigger than 500KHz): rmt_rx_channel_config_t rx_channel_cfg = {
.clk_src = RMT_CLK_SRC_XTAL,
.resolution_hz = 100000, // 100KHz
.mem_block_symbols = SOC_RMT_MEM_WORDS_PER_CHANNEL,
.gpio_num = TEST_RMT_GPIO_NUM_A,
}; Then the receive signal range should be achieved: rmt_receive_config_t rx_config = {
.signal_range_min_ns = 420000, // 420us
.signal_range_max_ns = 1400000, // 1.4ms
}; |
@suda-morris Thank you! I know there are a lot of peripheral that depend on those clocks, so I didn't know where I could make changes that didn't cause something to be broken down the road. And being able to (easily) modify the source is a big reason why I made the push to move away from Arduino. And again, I've gotten pulled from this to do something unrelated. I was able to make the changes and saw the changes reflected in the TX channel. I've tied my RX to ground and it is still telling me it is receiving bits, but that is a problem for another time. As of right now, @suda-morris has helped me achieve a lower group clock. Thank you! Also, thank you @atanisoft for helping me and raising visibility on this issue. Thank you, again! |
Answers checklist.
General issue report
I have an ESP32-S3-WROOM-1 and I am using ESP-IDF:
I (27) boot: ESP-IDF v5.3-dirty 2nd stage bootloader
I (27) boot: compile time Oct 21 2024 10:56:10
I (27) boot: Multicore bootloader
I (30) boot: chip revision: v0.2
I (34) boot.esp32s3: Boot SPI Speed : 80MHz
I (39) boot.esp32s3: SPI Mode : DIO
I (44) boot.esp32s3: SPI Flash Size : 8MB
...
I (199) cpu_start: Pro cpu start user code
I (203) cpu_start: cpu freq: 160000000 Hz
I (214) app_init: Application information:
I (219) app_init: Project name: rf_poc
I (224) app_init: App version: 1
I (228) app_init: Compile time: Oct 21 2024 10:56:03
I (234) app_init: ELF file SHA256: 171965faf...
I (240) app_init: ESP-IDF: v5.3-dirty
I (251) efuse_init: Min chip rev: v0.0
I (255) efuse_init: Max chip rev: v0.99
I (260) efuse_init: Chip rev: v0.2
The signals I want to receive will be at minimum 420uS (420000nS) and maximum 1.4mS (1400000nS). The only RMT clocks available are 80MHz APB clock, ~17.5MHz RC Fast, or external XTAL.
When I do the calculations with the 80MHz APB clock, my filter_reg_value is waaaay over the allowed 8-bit value. And when I used the 17.5MHz RC Fast clock, it brought the filter_reg_value down closer to an 8-bit number.
So, I'm going to plug in 255 for my filter_reg_value and solve for the 'filter_clock_resolution_hz', which is 607142.8Hz (607.1428kHz). This tell me that if can use a 500kHz clock, my filter_reg_value should be 210, and within the 8-bit range.
Am I missing something? Can I use a 500kHz as my XTAL? Can I use LEDC to produce the clock on a GPIO connected to the XTAL inputs?
Even a clock of 1MHz would give me the filter_reg_value of 255uS. Seems the RMT with the available clocks can't do what I need?
The text was updated successfully, but these errors were encountered: