forked from nrfconnect/sdk-nrf
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtimed_led_toggle.c
102 lines (80 loc) · 3.05 KB
/
timed_led_toggle.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/*
* Copyright (c) 2024 Nordic Semiconductor ASA
*
* SPDX-License-Identifier: LicenseRef-Nordic-5-Clause
*/
/** This file implements timed toggling a LED
*
* To achieve accurate toggling, it interacts with the controller
* clock using PPI.
*/
#include <zephyr/kernel.h>
#include <zephyr/device.h>
#include <zephyr/drivers/gpio.h>
#include <soc.h>
#include <nrfx_gpiote.h>
#include <helpers/nrfx_gppi.h>
#include "iso_time_sync.h"
#define GPIOTE_INST NRF_DT_GPIOTE_INST(DT_ALIAS(led1), gpios)
#define GPIOTE_NODE DT_NODELABEL(_CONCAT(gpiote, GPIOTE_INST))
#define LED_PIN NRF_DT_GPIOS_TO_PSEL(DT_ALIAS(led1), gpios)
BUILD_ASSERT(IS_ENABLED(_CONCAT(CONFIG_, _CONCAT(NRFX_GPIOTE, GPIOTE_INST))),
"NRFX_GPIOTE" STRINGIFY(GPIOTE_INST) " must be enabled in Kconfig");
static const nrfx_gpiote_t gpiote = NRFX_GPIOTE_INSTANCE(GPIOTE_INST);
static struct gpio_dt_spec led = GPIO_DT_SPEC_GET_OR(DT_ALIAS(led1), gpios, {0});
static uint8_t previous_led_value;
int timed_led_toggle_init(void)
{
int err;
uint8_t ppi_chan_led_toggle;
uint8_t gpiote_chan_led_toggle;
const nrfx_gpiote_output_config_t gpiote_output_cfg = NRFX_GPIOTE_DEFAULT_OUTPUT_CONFIG;
err = gpio_pin_configure_dt(&led, GPIO_OUTPUT_INACTIVE);
if (err != 0) {
printk("Error %d: failed to configure LED device %s pin %d\n", err, led.port->name,
led.pin);
return err;
}
if (nrfx_gpiote_channel_alloc(&gpiote, &gpiote_chan_led_toggle) != NRFX_SUCCESS) {
printk("Failed allocating GPIOTE chan for setting led\n");
return -ENOMEM;
}
const nrfx_gpiote_task_config_t task_cfg_led_toggle = {
.task_ch = gpiote_chan_led_toggle,
.polarity = NRF_GPIOTE_POLARITY_TOGGLE,
.init_val = (led.dt_flags & GPIO_ACTIVE_LOW) ?
NRF_GPIOTE_INITIAL_VALUE_HIGH : NRF_GPIOTE_INITIAL_VALUE_LOW,
};
if (nrfx_gpiote_output_configure(&gpiote, LED_PIN, &gpiote_output_cfg,
&task_cfg_led_toggle) != NRFX_SUCCESS) {
printk("Failed configuring GPIOTE chan for toggling led\n");
return -ENOMEM;
}
if (nrfx_gppi_channel_alloc(&ppi_chan_led_toggle) != NRFX_SUCCESS) {
printk("Failed allocating PPI chan for toggling led\n");
return -ENOMEM;
}
nrfx_gppi_channel_endpoints_setup(ppi_chan_led_toggle,
controller_time_trigger_event_addr_get(),
nrfx_gpiote_out_task_address_get(&gpiote, LED_PIN));
nrfx_gppi_channels_enable(BIT(ppi_chan_led_toggle));
nrfx_gpiote_out_task_enable(&gpiote, LED_PIN);
return 0;
}
void timed_led_toggle_trigger_at(uint8_t value, uint32_t timestamp_us)
{
if (value == previous_led_value) {
/* No need to set up trigger, as a trigger has already been configured. */
return;
}
/* First obtain the full 64-bit time. */
const uint64_t current_time_us = controller_time_us_get();
const uint64_t current_time_most_significant_word = current_time_us & 0xFFFFFFFF00000000UL;
uint64_t full_timestamp_us = current_time_most_significant_word | timestamp_us;
if (timestamp_us < (current_time_us & UINT32_MAX)) {
/* Trigger time is after UINT32 wrap */
full_timestamp_us += 0x100000000UL;
}
controller_time_trigger_set(full_timestamp_us);
previous_led_value = value;
}