Select one of the symbols to view example projects that use it.
 
Outline
#include "hardware/claim.h"
#include "hardware/pio.h"
#include "hardware/pio_instructions.h"
claimed
pio_sm_claim(PIO, uint)
pio_claim_sm_mask(PIO, uint)
pio_sm_unclaim(PIO, uint)
pio_claim_unused_sm(PIO, bool)
pio_sm_is_claimed(PIO, uint)
_used_instruction_space
find_offset_for_program(PIO, const pio_program_t *)
pio_set_gpio_base_unsafe(PIO, uint)
pio_set_gpio_base(PIO, uint)
is_gpio_compatible(PIO, uint32_t)
is_program_gpio_compatible(PIO, const pio_program_t *)
add_program_at_offset_check(PIO, const pio_program_t *, uint)
pio_can_add_program(PIO, const pio_program_t *)
pio_can_add_program_at_offset(PIO, const pio_program_t *, uint)
add_program_at_offset(PIO, const pio_program_t *, uint)
pio_add_program(PIO, const pio_program_t *)
pio_add_program_at_offset(PIO, const pio_program_t *, uint)
pio_remove_program(PIO, const pio_program_t *, uint)
pio_clear_instruction_memory(PIO)
pio_sm_set_pins(PIO, uint, uint32_t)
pio_sm_set_pins_with_mask(PIO, uint, uint32_t, uint32_t)
pio_sm_set_pindirs_with_mask(PIO, uint, uint32_t, uint32_t)
pio_sm_set_consecutive_pindirs(PIO, uint, uint, uint, bool)
pio_sm_init(PIO, uint, uint, const pio_sm_config *)
pio_sm_drain_tx_fifo(PIO, uint)
pio_claim_free_sm_and_add_program(const pio_program_t *, PIO *, uint *, uint *)
pio_claim_free_sm_and_add_program_for_gpio_range(const pio_program_t *, PIO *, uint *, uint *, uint, uint, bool)
pio_remove_program_and_unclaim_sm(const pio_program_t *, PIO, uint, uint)
Files
loading...
SourceVuRaspberry Pi Pico SDK and ExamplesPicoSDKsrc/rp2_common/hardware_pio/pio.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
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * Copyright (c) 2020 Raspberry Pi (Trading) Ltd. * * SPDX-License-Identifier: BSD-3-Clause *//* ... */ #include "hardware/claim.h" #include "hardware/pio.h" #include "hardware/pio_instructions.h" // sanity check check_hw_layout(pio_hw_t, sm[0].clkdiv, PIO_SM0_CLKDIV_OFFSET); check_hw_layout(pio_hw_t, sm[1].clkdiv, PIO_SM1_CLKDIV_OFFSET); check_hw_layout(pio_hw_t, instr_mem[0], PIO_INSTR_MEM0_OFFSET); check_hw_layout(pio_hw_t, inte0, PIO_IRQ0_INTE_OFFSET); check_hw_layout(pio_hw_t, irq_ctrl[0].inte, PIO_IRQ0_INTE_OFFSET); check_hw_layout(pio_hw_t, txf[1], PIO_TXF1_OFFSET); check_hw_layout(pio_hw_t, rxf[3], PIO_RXF3_OFFSET); check_hw_layout(pio_hw_t, ints1, PIO_IRQ1_INTS_OFFSET); check_hw_layout(pio_hw_t, irq_ctrl[1].ints, PIO_IRQ1_INTS_OFFSET); static uint8_t claimed[(NUM_PIO_STATE_MACHINES * NUM_PIOS + 7) >> 3]; void pio_sm_claim(PIO pio, uint sm) { check_sm_param(sm); uint which = pio_get_index(pio); const char *msg = #if PICO_PIO_VERSION > 0 which == 2 ? "PIO 2 SM (%d - 8) already claimed" : #endif which == 1 ? "PIO 1 SM (%d - 4) already claimed" : "PIO 0 SM %d already claimed"; hw_claim_or_assert(&claimed[0], which * NUM_PIO_STATE_MACHINES + sm, msg); }{ ... } void pio_claim_sm_mask(PIO pio, uint sm_mask) { for(uint i = 0; sm_mask; i++, sm_mask >>= 1u) { if (sm_mask & 1u) pio_sm_claim(pio, i); }for (uint i = 0; sm_mask; i++, sm_mask >>= 1u) { ... } }{ ... } void pio_sm_unclaim(PIO pio, uint sm) { check_sm_param(sm); uint which = pio_get_index(pio); hw_claim_clear(&claimed[0], which * NUM_PIO_STATE_MACHINES + sm); }{ ... } int pio_claim_unused_sm(PIO pio, bool required) { // PIO index ranges from 0 to NUM_PIOS - 1. uint which = pio_get_index(pio); uint base = which * NUM_PIO_STATE_MACHINES; int index = hw_claim_unused_from_range((uint8_t*)&claimed[0], required, base, base + NUM_PIO_STATE_MACHINES - 1, "No PIO state machines are available"); return index >= (int)base ? index - (int)base : -1; }{ ... } bool pio_sm_is_claimed(PIO pio, uint sm) { check_sm_param(sm); uint which = pio_get_index(pio); return hw_is_claimed(&claimed[0], which * NUM_PIO_STATE_MACHINES + sm); }{ ... } static_assert(PIO_INSTRUCTION_COUNT <= 32, ""); static uint32_t _used_instruction_space[NUM_PIOS]; static int find_offset_for_program(PIO pio, const pio_program_t *program) { assert(program->length <= PIO_INSTRUCTION_COUNT); uint32_t used_mask = _used_instruction_space[pio_get_index(pio)]; uint32_t program_mask = (1u << program->length) - 1; if (program->origin >= 0) { if (program->origin > 32 - program->length) return PICO_ERROR_GENERIC; return used_mask & (program_mask << program->origin) ? -1 : program->origin; }if (program->origin >= 0) { ... } else { // work down from the top always for (int i = 32 - program->length; i >= 0; i--) { if (!(used_mask & (program_mask << (uint) i))) { return i; }if (!(used_mask & (program_mask << (uint) i))) { ... } }for (int i = 32 - program->length; i >= 0; i--) { ... } return PICO_ERROR_INSUFFICIENT_RESOURCES; }else { ... } }{ ... } static int pio_set_gpio_base_unsafe(PIO pio, uint gpio_base) { invalid_params_if_and_return(PIO, gpio_base != 0 && (!PICO_PIO_VERSION || gpio_base != 16), PICO_ERROR_BAD_ALIGNMENT); #if PICO_PIO_VERSION > 0 uint32_t used_mask = _used_instruction_space[pio_get_index(pio)]; invalid_params_if_and_return(PIO, used_mask, PICO_ERROR_INVALID_STATE); pio->gpiobase = gpio_base;/* ... */ #else ((void)pio); ((void)gpio_base);/* ... */ #endif return PICO_OK; }{ ... } int pio_set_gpio_base(PIO pio, uint gpio_base) { int rc = PICO_OK; #if PICO_PIO_VERSION > 0 uint32_t save = hw_claim_lock(); rc = pio_set_gpio_base_unsafe(pio, gpio_base); hw_claim_unlock(save);/* ... */ #else ((void)pio); ((void)gpio_base);/* ... */ #endif return rc; }{ ... } static bool is_gpio_compatible(PIO pio, uint32_t used_gpio_ranges) { #if PICO_PIO_VERSION > 0 bool gpio_base = pio_get_gpio_base(pio); return !((gpio_base && (used_gpio_ranges & 1)) || (!gpio_base && (used_gpio_ranges & 4)));/* ... */ #else ((void)pio); ((void)used_gpio_ranges); return true;/* ... */ #endif }{ ... } static bool is_program_gpio_compatible(PIO pio, const pio_program_t *program) { #if PICO_PIO_VERSION > 0 return is_gpio_compatible(pio, program->used_gpio_ranges); #else ((void)pio); ((void)program); return true;/* ... */ #endif }{ ... } static int add_program_at_offset_check(PIO pio, const pio_program_t *program, uint offset) { valid_params_if(HARDWARE_PIO, offset < PIO_INSTRUCTION_COUNT); valid_params_if(HARDWARE_PIO, offset + program->length <= PIO_INSTRUCTION_COUNT); #if PICO_PIO_VERSION == 0 if (program->pio_version) return PICO_ERROR_VERSION_MISMATCH; #endif if (!is_program_gpio_compatible(pio, program)) return PICO_ERROR_BAD_ALIGNMENT; // todo better error? if (program->origin >= 0 && (uint)program->origin != offset) return PICO_ERROR_BAD_ALIGNMENT; // todo better error? uint32_t used_mask = _used_instruction_space[pio_get_index(pio)]; uint32_t program_mask = (1u << program->length) - 1; return (used_mask & (program_mask << offset)) ? PICO_ERROR_INSUFFICIENT_RESOURCES : PICO_OK; }{ ... } bool pio_can_add_program(PIO pio, const pio_program_t *program) { uint32_t save = hw_claim_lock(); int rc = find_offset_for_program(pio, program); if (rc >= 0) rc = add_program_at_offset_check(pio, program, (uint)rc); hw_claim_unlock(save); return rc == 0; }{ ... } bool pio_can_add_program_at_offset(PIO pio, const pio_program_t *program, uint offset) { uint32_t save = hw_claim_lock(); bool rc = add_program_at_offset_check(pio, program, offset) == 0; hw_claim_unlock(save); return rc; }{ ... } static int add_program_at_offset(PIO pio, const pio_program_t *program, uint offset) { int rc = add_program_at_offset_check(pio, program, offset); if (rc != 0) return rc; for (uint i = 0; i < program->length; ++i) { uint16_t instr = program->instructions[i]; pio->instr_mem[offset + i] = pio_instr_bits_jmp != _pio_major_instr_bits(instr) ? instr : instr + offset; }for (uint i = 0; i < program->length; ++i) { ... } uint32_t program_mask = (1u << program->length) - 1; _used_instruction_space[pio_get_index(pio)] |= program_mask << offset; return (int)offset; }{ ... } // these assert if unable int pio_add_program(PIO pio, const pio_program_t *program) { uint32_t save = hw_claim_lock(); int offset = find_offset_for_program(pio, program); if (offset >= 0) { offset = add_program_at_offset(pio, program, (uint) offset); }if (offset >= 0) { ... } hw_claim_unlock(save); return offset; }{ ... } int pio_add_program_at_offset(PIO pio, const pio_program_t *program, uint offset) { uint32_t save = hw_claim_lock(); int rc = add_program_at_offset(pio, program, offset); hw_claim_unlock(save); return rc; }{ ... } void pio_remove_program(PIO pio, const pio_program_t *program, uint loaded_offset) { uint32_t program_mask = (1u << program->length) - 1; program_mask <<= loaded_offset; uint32_t save = hw_claim_lock(); assert(program_mask == (_used_instruction_space[pio_get_index(pio)] & program_mask)); _used_instruction_space[pio_get_index(pio)] &= ~program_mask; hw_claim_unlock(save); }{ ... } void pio_clear_instruction_memory(PIO pio) { uint32_t save = hw_claim_lock(); _used_instruction_space[pio_get_index(pio)] = 0; for(uint i=0;i<PIO_INSTRUCTION_COUNT;i++) { pio->instr_mem[i] = pio_encode_jmp(i); }for (uint i=0;i hw_claim_unlock(save); }{ ... } // Set the value of all PIO pins. This is done by forcibly executing // instructions on a "victim" state machine, sm. Ideally you should choose one // which is not currently running a program. This is intended for one-time // setup of initial pin states. void pio_sm_set_pins(PIO pio, uint sm, uint32_t pins) { check_pio_param(pio); check_sm_param(sm); uint32_t pinctrl_saved = pio->sm[sm].pinctrl; uint32_t execctrl_saved = pio->sm[sm].execctrl; hw_clear_bits(&pio->sm[sm].execctrl, 1u << PIO_SM0_EXECCTRL_OUT_STICKY_LSB); uint remaining = 32; uint base = 0; while (remaining) { uint decrement = remaining > 5 ? 5 : remaining; pio->sm[sm].pinctrl = (decrement << PIO_SM0_PINCTRL_SET_COUNT_LSB) | (base << PIO_SM0_PINCTRL_SET_BASE_LSB); pio_sm_exec(pio, sm, pio_encode_set(pio_pins, pins & 0x1fu)); remaining -= decrement; base += decrement; pins >>= 5; }while (remaining) { ... } pio->sm[sm].pinctrl = pinctrl_saved; pio->sm[sm].execctrl = execctrl_saved; }{ ... } void pio_sm_set_pins_with_mask(PIO pio, uint sm, uint32_t pinvals, uint32_t pin_mask) { check_pio_param(pio); check_sm_param(sm); uint32_t pinctrl_saved = pio->sm[sm].pinctrl; uint32_t execctrl_saved = pio->sm[sm].execctrl; hw_clear_bits(&pio->sm[sm].execctrl, 1u << PIO_SM0_EXECCTRL_OUT_STICKY_LSB); while (pin_mask) { uint base = (uint)__builtin_ctz(pin_mask); pio->sm[sm].pinctrl = (1u << PIO_SM0_PINCTRL_SET_COUNT_LSB) | (base << PIO_SM0_PINCTRL_SET_BASE_LSB); pio_sm_exec(pio, sm, pio_encode_set(pio_pins, (pinvals >> base) & 0x1u)); pin_mask &= pin_mask - 1; }while (pin_mask) { ... } pio->sm[sm].pinctrl = pinctrl_saved; pio->sm[sm].execctrl = execctrl_saved; }{ ... } void pio_sm_set_pindirs_with_mask(PIO pio, uint sm, uint32_t pindirs, uint32_t pin_mask) { check_pio_param(pio); check_sm_param(sm); uint32_t pinctrl_saved = pio->sm[sm].pinctrl; uint32_t execctrl_saved = pio->sm[sm].execctrl; hw_clear_bits(&pio->sm[sm].execctrl, 1u << PIO_SM0_EXECCTRL_OUT_STICKY_LSB); while (pin_mask) { uint base = (uint)__builtin_ctz(pin_mask); pio->sm[sm].pinctrl = (1u << PIO_SM0_PINCTRL_SET_COUNT_LSB) | (base << PIO_SM0_PINCTRL_SET_BASE_LSB); pio_sm_exec(pio, sm, pio_encode_set(pio_pindirs, (pindirs >> base) & 0x1u)); pin_mask &= pin_mask - 1; }while (pin_mask) { ... } pio->sm[sm].pinctrl = pinctrl_saved; pio->sm[sm].execctrl = execctrl_saved; }{ ... } int pio_sm_set_consecutive_pindirs(PIO pio, uint sm, uint pin, uint count, bool is_out) { check_pio_param(pio); check_sm_param(sm); pin -= pio_get_gpio_base(pio); invalid_params_if_and_return(PIO, pin >= 32u, PICO_ERROR_INVALID_ARG); uint32_t pinctrl_saved = pio->sm[sm].pinctrl; uint32_t execctrl_saved = pio->sm[sm].execctrl; hw_clear_bits(&pio->sm[sm].execctrl, 1u << PIO_SM0_EXECCTRL_OUT_STICKY_LSB); uint pindir_val = is_out ? 0x1f : 0; while (count > 5) { pio->sm[sm].pinctrl = (5u << PIO_SM0_PINCTRL_SET_COUNT_LSB) | (pin << PIO_SM0_PINCTRL_SET_BASE_LSB); pio_sm_exec(pio, sm, pio_encode_set(pio_pindirs, pindir_val)); count -= 5; pin = (pin + 5) & 0x1f; }while (count > 5) { ... } pio->sm[sm].pinctrl = (count << PIO_SM0_PINCTRL_SET_COUNT_LSB) | (pin << PIO_SM0_PINCTRL_SET_BASE_LSB); pio_sm_exec(pio, sm, pio_encode_set(pio_pindirs, pindir_val)); pio->sm[sm].pinctrl = pinctrl_saved; pio->sm[sm].execctrl = execctrl_saved; return PICO_OK; }{ ... } int pio_sm_init(PIO pio, uint sm, uint initial_pc, const pio_sm_config *config) { valid_params_if(HARDWARE_PIO, initial_pc < PIO_INSTRUCTION_COUNT); // Halt the machine, set some sensible defaults pio_sm_set_enabled(pio, sm, false); int rc; if (config) { rc = pio_sm_set_config(pio, sm, config); }if (config) { ... } else { pio_sm_config c = pio_get_default_sm_config(); rc = pio_sm_set_config(pio, sm, &c); }else { ... } if (rc) return rc; pio_sm_clear_fifos(pio, sm); // Clear FIFO debug flags const uint32_t fdebug_sm_mask = (1u << PIO_FDEBUG_TXOVER_LSB) | (1u << PIO_FDEBUG_RXUNDER_LSB) | (1u << PIO_FDEBUG_TXSTALL_LSB) | (1u << PIO_FDEBUG_RXSTALL_LSB); pio->fdebug = fdebug_sm_mask << sm; // Finally, clear some internal SM state pio_sm_restart(pio, sm); pio_sm_clkdiv_restart(pio, sm); pio_sm_exec(pio, sm, pio_encode_jmp(initial_pc)); return PICO_OK; }{ ... } void pio_sm_drain_tx_fifo(PIO pio, uint sm) { uint instr = (pio->sm[sm].shiftctrl & PIO_SM0_SHIFTCTRL_AUTOPULL_BITS) ? pio_encode_out(pio_null, 32) : pio_encode_pull(false, false); while (!pio_sm_is_tx_fifo_empty(pio, sm)) { pio_sm_exec(pio, sm, instr); }while (!pio_sm_is_tx_fifo_empty(pio, sm)) { ... } }{ ... } bool pio_claim_free_sm_and_add_program(const pio_program_t *program, PIO *pio, uint *sm, uint *offset) { return pio_claim_free_sm_and_add_program_for_gpio_range(program, pio, sm, offset, 0, 0, false); }{ ... } bool pio_claim_free_sm_and_add_program_for_gpio_range(const pio_program_t *program, PIO *pio, uint *sm, uint *offset, uint gpio_base, uint gpio_count, bool set_gpio_base) { invalid_params_if(HARDWARE_PIO, (gpio_base + gpio_count) > NUM_BANK0_GPIOS); #if !PICO_PIO_USE_GPIO_BASE // short-circuit some logic when not using GIO_BASE set_gpio_base = 0; gpio_count = 0;/* ... */ #endif // note if we gpio_count == 0, we don't care about GPIOs so use a zero mask for what we require // if gpio_count > 0, then we just set used mask for the ends, since that is all that is checked at the moment uint32_t required_gpio_ranges; if (gpio_count) required_gpio_ranges = (1u << (gpio_base >> 4)) | (1u << ((gpio_base + gpio_count - 1) >> 4)); else required_gpio_ranges = 0; int passes = set_gpio_base ? 2 : 1; for(int pass = 0; pass < passes; pass++) { int pio_num = NUM_PIOS; while (pio_num--) { *pio = pio_get_instance((uint)pio_num); // We need to claim an SM on the PIO int8_t sm_index[NUM_PIO_STATE_MACHINES]; // on second pass, if there is one, we try and claim all the state machines so that we can change the GPIO base uint num_claimed; for(num_claimed = 0; num_claimed < (pass ? NUM_PIO_STATE_MACHINES : 1u) ; num_claimed++) { sm_index[num_claimed] = (int8_t)pio_claim_unused_sm(*pio, false); if (sm_index[num_claimed] < 0) break; }for (num_claimed = 0; num_claimed < (pass ? NUM_PIO_STATE_MACHINES : 1u) ; num_claimed++) { ... } if (num_claimed && (!pass || num_claimed == NUM_PIO_STATE_MACHINES)) { uint32_t save = hw_claim_lock(); if (pass) { pio_set_gpio_base_unsafe(*pio, required_gpio_ranges & 4 ? 16 : 0); }if (pass) { ... } int rc = is_gpio_compatible(*pio, required_gpio_ranges) ? PICO_OK : PICO_ERROR_BAD_ALIGNMENT; if (rc == PICO_OK) rc = find_offset_for_program(*pio, program); if (rc >= 0) rc = add_program_at_offset(*pio, program, (uint)rc); if (rc >= 0) { *sm = (uint) sm_index[0]; *offset = (uint) rc; }if (rc >= 0) { ... } hw_claim_unlock(save); // always un-claim all SMs other than the one we need (array index 0), // or all of them if we had an error for (uint i = (rc >= 0); i < num_claimed; i++) { pio_sm_unclaim(*pio, (uint) sm_index[i]); }for (uint i = (rc >= 0); i < num_claimed; i++) { ... } if (rc >= 0) { return true; }if (rc >= 0) { ... } }if (num_claimed && (!pass || num_claimed == NUM_PIO_STATE_MACHINES)) { ... } }while (pio_num--) { ... } }for (int pass = 0; pass < passes; pass++) { ... } *pio = NULL; return false; }{ ... } void pio_remove_program_and_unclaim_sm(const pio_program_t *program, PIO pio, uint sm, uint offset) { check_pio_param(pio); check_sm_param(sm); pio_remove_program(pio, program, offset); pio_sm_unclaim(pio, sm); }{ ... }
Details
Show:
from
Types: Columns:
This file uses the notable symbols shown below. Click anywhere in the file to view more details.