Select one of the symbols to view example projects that use it.
 
Outline
#include <stdlib.h>
#include <string.h>
#include <sys/lock.h>
#include "esp_pm.h"
#include "esp_system.h"
#include "sys/queue.h"
#include "freertos/FreeRTOS.h"
#include "esp_private/pm_impl.h"
#include "esp_timer.h"
#include "sdkconfig.h"
esp_pm_lock
s_lock_type_names
s_list
s_list_lock
esp_pm_lock_create(esp_pm_lock_type_t, int, const char *, esp_pm_lock_handle_t *)
esp_pm_lock_delete(esp_pm_lock_handle_t)
esp_pm_lock_acquire(esp_pm_lock_handle_t)
esp_pm_lock_release(esp_pm_lock_handle_t)
esp_pm_dump_locks(FILE *)
Files
loading...
SourceVuESP-IDF Framework and ExamplesESP-IDFcomponents/esp_pm/pm_locks.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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * SPDX-FileCopyrightText: 2016-2023 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 *//* ... */ #include <stdlib.h> #include <string.h> #include <sys/lock.h> #include "esp_pm.h" #include "esp_system.h" #include "sys/queue.h" #include "freertos/FreeRTOS.h" #include "esp_private/pm_impl.h" #include "esp_timer.h" #include "sdkconfig.h"10 includes typedef struct esp_pm_lock { esp_pm_lock_type_t type; /*!< type passed to esp_pm_lock_create */ int arg; /*!< argument passed to esp_pm_lock_create */ pm_mode_t mode; /*!< implementation-defined mode for this type of lock*/ const char* name; /*!< used to identify the lock */ SLIST_ENTRY(esp_pm_lock) next; /*!< linked list pointer */ size_t count; /*!< lock count */ portMUX_TYPE spinlock; /*!< spinlock used when operating on 'count' */ #ifdef WITH_PROFILING pm_time_t last_taken; /*!< time what the lock was taken (valid if count > 0) */ pm_time_t time_held; /*!< total time the lock was taken. If count > 0, this doesn't include the time since last_taken *//* ... */ size_t times_taken; /*!< number of times the lock was ever taken *//* ... */ #endif }{ ... } esp_pm_lock_t; static const char* s_lock_type_names[] = { "CPU_FREQ_MAX", "APB_FREQ_MAX", "NO_LIGHT_SLEEP" }{...}; /* List of all existing locks, used for esp_pm_dump_locks */ static SLIST_HEAD(esp_pm_locks_head, esp_pm_lock) s_list = SLIST_HEAD_INITIALIZER(s_head); /* Protects the above list */ static _lock_t s_list_lock; esp_err_t esp_pm_lock_create(esp_pm_lock_type_t lock_type, int arg, const char* name, esp_pm_lock_handle_t* out_handle) { #ifndef CONFIG_PM_ENABLE return ESP_ERR_NOT_SUPPORTED; #endif if (out_handle == NULL) { return ESP_ERR_INVALID_ARG; }{...} esp_pm_lock_t* new_lock = (esp_pm_lock_t*) calloc(1, sizeof(*new_lock)); if (!new_lock) { return ESP_ERR_NO_MEM; }{...} new_lock->type = lock_type; new_lock->arg = arg; new_lock->mode = esp_pm_impl_get_mode(lock_type, arg); new_lock->name = name; new_lock->spinlock = (portMUX_TYPE) portMUX_INITIALIZER_UNLOCKED; *out_handle = new_lock; _lock_acquire(&s_list_lock); SLIST_INSERT_HEAD(&s_list, new_lock, next); _lock_release(&s_list_lock); return ESP_OK; }{ ... } esp_err_t esp_pm_lock_delete(esp_pm_lock_handle_t handle) { #ifndef CONFIG_PM_ENABLE return ESP_ERR_NOT_SUPPORTED; #endif if (handle == NULL) { return ESP_ERR_INVALID_ARG; }{...} if (handle->count > 0) { return ESP_ERR_INVALID_STATE; }{...} _lock_acquire(&s_list_lock); SLIST_REMOVE(&s_list, handle, esp_pm_lock, next); _lock_release(&s_list_lock); free(handle); return ESP_OK; }{ ... } esp_err_t IRAM_ATTR esp_pm_lock_acquire(esp_pm_lock_handle_t handle) { #ifndef CONFIG_PM_ENABLE return ESP_ERR_NOT_SUPPORTED; #endif if (handle == NULL) { return ESP_ERR_INVALID_ARG; }{...} portENTER_CRITICAL_SAFE(&handle->spinlock); if (handle->count++ == 0) { pm_time_t now = 0; #ifdef WITH_PROFILING now = pm_get_time(); #endif esp_pm_impl_switch_mode(handle->mode, MODE_LOCK, now); #ifdef WITH_PROFILING handle->last_taken = now; handle->times_taken++;/* ... */ #endif }{...} portEXIT_CRITICAL_SAFE(&handle->spinlock); return ESP_OK; }{ ... } esp_err_t IRAM_ATTR esp_pm_lock_release(esp_pm_lock_handle_t handle) { #ifndef CONFIG_PM_ENABLE return ESP_ERR_NOT_SUPPORTED; #endif if (handle == NULL) { return ESP_ERR_INVALID_ARG; }{...} esp_err_t ret = ESP_OK; portENTER_CRITICAL_SAFE(&handle->spinlock); if (handle->count == 0) { ret = ESP_ERR_INVALID_STATE; goto out; }{...} if (--handle->count == 0) { pm_time_t now = 0; #ifdef WITH_PROFILING now = pm_get_time(); handle->time_held += now - handle->last_taken;/* ... */ #endif esp_pm_impl_switch_mode(handle->mode, MODE_UNLOCK, now); }{...} out: portEXIT_CRITICAL_SAFE(&handle->spinlock); return ret; }{ ... } esp_err_t esp_pm_dump_locks(FILE* stream) { #ifndef CONFIG_PM_ENABLE return ESP_ERR_NOT_SUPPORTED; #endif #ifdef WITH_PROFILING pm_time_t cur_time = pm_get_time(); pm_time_t cur_time_d100 = cur_time / 100;/* ... */ #endif // WITH_PROFILING _lock_acquire(&s_list_lock); #ifdef WITH_PROFILING fprintf(stream, "Time since bootup: %lld us\n", cur_time); #endif fprintf(stream, "Lock stats:\n"); #ifdef WITH_PROFILING fprintf(stream, "%-15s %-14s %-5s %-8s %-13s %-14s %-8s\n", "Name", "Type", "Arg", "Active", "Total_count", "Time(us)", "Time(%)");/* ... */ #else fprintf(stream, "%-15s %-14s %-5s %-8s\n", "Name", "Type", "Arg", "Active"); #endif esp_pm_lock_t* it; char line[128]; SLIST_FOREACH(it, &s_list, next) { char *buf = line; size_t len = sizeof(line); size_t cb; portENTER_CRITICAL(&it->spinlock); if (it->name == NULL) { cb = snprintf(buf, len, "lock@%p ", it); }{...} else { cb = snprintf(buf, len, "%-15.15s ", it->name); }{...} assert(cb <= len); // above formats should fit into sizeof(line) buf += cb; len -= cb; #ifdef WITH_PROFILING pm_time_t time_held = it->time_held; if (it->count > 0) { time_held += cur_time - it->last_taken; }{...} snprintf(buf, len, "%-14s %-5d %-8d %-13d %-14lld %-3lld%%\n", s_lock_type_names[it->type], it->arg, it->count, it->times_taken, time_held, (time_held + cur_time_d100 - 1) / cur_time_d100);/* ... */ #else snprintf(buf, len, "%-14s %-5d %-8d\n", s_lock_type_names[it->type], it->arg, it->count); #endif // WITH_PROFILING portEXIT_CRITICAL(&it->spinlock); fputs(line, stream); }{...} _lock_release(&s_list_lock); #ifdef WITH_PROFILING esp_pm_impl_dump_stats(stream); #endif return ESP_OK; }{ ... }
Details