]> sigrok.org Git - sigrok-cli.git/blob - decode.c
doc: update IRC reference to Libera.Chat
[sigrok-cli.git] / decode.c
1 /*
2  * This file is part of the sigrok-cli project.
3  *
4  * Copyright (C) 2013 Bert Vermeulen <bert@biot.com>
5  *
6  * This program is free software: you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation, either version 3 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19
20 #include <config.h>
21 #include <stdlib.h>
22 #include <string.h>
23 #include <glib.h>
24 #include "sigrok-cli.h"
25
26 #ifdef HAVE_SRD
27 static GHashTable *pd_ann_visible = NULL;
28 static GHashTable *pd_meta_visible = NULL;
29 static GHashTable *pd_binary_visible = NULL;
30 static GHashTable *pd_channel_maps = NULL;
31
32 uint64_t pd_samplerate = 0;
33
34 extern struct srd_session *srd_sess;
35
36 static int opts_to_gvar(struct srd_decoder *dec, GHashTable *hash,
37                 GHashTable **options)
38 {
39         struct srd_decoder_option *o;
40         GSList *optl;
41         GVariant *gvar;
42         gint64 val_int;
43         double val_dbl;
44         int ret;
45         char *val_str, *conv;
46
47         ret = TRUE;
48         *options = g_hash_table_new_full(g_str_hash, g_str_equal, g_free,
49                         (GDestroyNotify)g_variant_unref);
50
51         for (optl = dec->options; optl; optl = optl->next) {
52                 o = optl->data;
53                 if (!(val_str = g_hash_table_lookup(hash, o->id)))
54                         /* Not specified. */
55                         continue;
56                 if (g_variant_is_of_type(o->def, G_VARIANT_TYPE_STRING)) {
57                         gvar = g_variant_new_string(val_str);
58                 } else if (g_variant_is_of_type(o->def, G_VARIANT_TYPE_INT64)) {
59                         conv = NULL;
60                         val_int = strtoll(val_str, &conv, 0);
61                         if (!conv || conv == val_str || *conv) {
62                                 g_critical("Protocol decoder '%s' option '%s' "
63                                                 "requires a number.", dec->name, o->id);
64                                 ret = FALSE;
65                                 break;
66                         }
67                         gvar = g_variant_new_int64(val_int);
68                 } else if (g_variant_is_of_type(o->def, G_VARIANT_TYPE_DOUBLE)) {
69                         conv = NULL;
70                         val_dbl = strtod(val_str, &conv);
71                         if (!conv || conv == val_str || *conv) {
72                                 g_critical("Protocol decoder '%s' option '%s' requires a float number.",
73                                         dec->name, o->id);
74                                 ret = FALSE;
75                                 break;
76                         }
77                         gvar = g_variant_new_double(val_dbl);
78                 } else {
79                         g_critical("Unsupported type for option '%s' (%s)",
80                                         o->id, g_variant_get_type_string(o->def));
81                         ret = FALSE;
82                         break;
83                 }
84                 g_variant_ref_sink(gvar);
85                 g_hash_table_insert(*options, g_strdup(o->id), gvar);
86                 g_hash_table_remove(hash, o->id);
87         }
88
89         return ret;
90 }
91
92 static int move_hash_element(GHashTable *src, GHashTable *dest, void *key)
93 {
94         void *orig_key, *value;
95
96         if (!g_hash_table_lookup_extended(src, key, &orig_key, &value))
97                 /* Not specified. */
98                 return FALSE;
99         g_hash_table_steal(src, orig_key);
100         g_hash_table_insert(dest, orig_key, value);
101
102         return TRUE;
103 }
104
105 static GHashTable *extract_channel_map(struct srd_decoder *dec, GHashTable *hash)
106 {
107         GHashTable *channel_map;
108         struct srd_channel *pdch;
109         GSList *l;
110
111         channel_map = g_hash_table_new_full(g_str_hash, g_str_equal,
112                                           g_free, g_free);
113
114         for (l = dec->channels; l; l = l->next) {
115                 pdch = l->data;
116                 move_hash_element(hash, channel_map, pdch->id);
117         }
118         for (l = dec->opt_channels; l; l = l->next) {
119                 pdch = l->data;
120                 move_hash_element(hash, channel_map, pdch->id);
121         }
122
123         return channel_map;
124 }
125
126 static int register_pd(char *opt_pds, char *opt_pd_annotations)
127 {
128         int ret;
129         struct srd_decoder *dec;
130         struct srd_decoder_inst *di, *di_prior;
131         char **pdtokens, **pdtok, *pd_name;
132         GHashTable *pd_opthash, *options, *channels;
133         GList *leftover, *l;
134
135         ret = 0;
136         pd_name = NULL;
137         di_prior = NULL;
138         pd_opthash = options = channels = NULL;
139
140         pdtokens = g_strsplit(opt_pds, ",", 0);
141         for (pdtok = pdtokens; *pdtok; pdtok++) {
142                 if (!(pd_opthash = parse_generic_arg(*pdtok, TRUE, NULL))) {
143                         g_critical("Invalid protocol decoder option '%s'.", *pdtok);
144                         break;
145                 }
146
147                 pd_name = g_strdup(g_hash_table_lookup(pd_opthash, "sigrok_key"));
148                 g_hash_table_remove(pd_opthash, "sigrok_key");
149                 if (srd_decoder_load(pd_name) != SRD_OK) {
150                         g_critical("Failed to load protocol decoder %s.", pd_name);
151                         ret = 1;
152                         break;
153                 }
154                 if (!(dec = srd_decoder_get_by_id(pd_name))) {
155                         g_critical("Failed to get decoder %s by id.", pd_name);
156                         ret = 1;
157                         break;
158                 }
159
160                 /* Convert decoder option and channel values to GVariant. */
161                 if (!opts_to_gvar(dec, pd_opthash, &options)) {
162                         ret = 1;
163                         break;
164                 }
165                 channels = extract_channel_map(dec, pd_opthash);
166
167                 if (g_hash_table_size(pd_opthash) > 0) {
168                         leftover = g_hash_table_get_keys(pd_opthash);
169                         for (l = leftover; l; l = l->next)
170                                 g_critical("Unknown option or channel '%s'", (char *)l->data);
171                         g_list_free(leftover);
172                         break;
173                 }
174
175                 if (!(di = srd_inst_new(srd_sess, pd_name, options))) {
176                         g_critical("Failed to instantiate protocol decoder %s.", pd_name);
177                         ret = 1;
178                         break;
179                 }
180
181                 if (pdtok == pdtokens) {
182                         /*
183                          * Save the channel setup for later, but only on the
184                          * first decoder (stacked decoders don't get channels).
185                          */
186                         g_hash_table_insert(pd_channel_maps, g_strdup(di->inst_id), channels);
187                         channels = NULL;
188                 }
189
190                 /*
191                  * If no annotation list was specified, add them all in now.
192                  * This will be pared down later to leave only the last PD
193                  * in the stack.
194                  */
195                 if (!opt_pd_annotations) {
196                         g_hash_table_insert(pd_ann_visible, g_strdup(di->decoder->id),
197                                         g_slist_append(NULL, GINT_TO_POINTER(-1)));
198                 }
199                 if (di_prior) {
200                         if (srd_inst_stack(srd_sess, di_prior, di) != SRD_OK) {
201                                 g_critical("Failed to stack %s -> %s.",
202                                         di_prior->inst_id, di->inst_id);
203                                 ret = 1;
204                                 break;
205                         }
206                         /* Remove annotations from prior levels. */
207                         if (!opt_pd_annotations)
208                                 g_hash_table_remove(pd_ann_visible, di_prior->inst_id);
209                 }
210                 di_prior = di;
211                 g_free(pd_name);
212                 g_hash_table_destroy(pd_opthash);
213                 g_hash_table_destroy(options);
214                 pd_opthash = options = NULL;
215         }
216
217         if (pd_opthash)
218                 g_hash_table_destroy(pd_opthash);
219         if (options)
220                 g_hash_table_destroy(options);
221         if (channels)
222                 g_hash_table_destroy(channels);
223
224         g_strfreev(pdtokens);
225
226         return ret;
227 }
228
229 /*
230  * Register all the PDs from all stacks.
231  *
232  * Each PD string is a single stack such as "uart:baudrate=19200,modbus".
233  */
234 int register_pds(gchar **all_pds, char *opt_pd_annotations)
235 {
236         int ret;
237
238         ret = 0;
239         pd_ann_visible = g_hash_table_new_full(g_str_hash, g_str_equal,
240                                                g_free, NULL);
241         pd_channel_maps = g_hash_table_new_full(g_str_hash,
242                 g_str_equal, g_free, (GDestroyNotify)g_hash_table_destroy);
243
244         for (int i = 0; all_pds[i]; i++)
245                 ret += register_pd(all_pds[i], opt_pd_annotations);
246
247         return ret;
248 }
249
250 static void map_pd_inst_channels(void *key, void *value, void *user_data)
251 {
252         GHashTable *channel_map;
253         GHashTable *channel_indices;
254         GSList *channel_list;
255         struct srd_decoder_inst *di;
256         GVariant *var;
257         void *channel_id;
258         void *channel_target;
259         struct sr_channel *ch;
260         GHashTableIter iter;
261
262         channel_map = value;
263         channel_list = user_data;
264
265         di = srd_inst_find_by_id(srd_sess, key);
266         if (!di) {
267                 g_critical("Protocol decoder instance \"%s\" not found.",
268                            (char *)key);
269                 return;
270         }
271         channel_indices = g_hash_table_new_full(g_str_hash, g_str_equal, g_free,
272                                               (GDestroyNotify)g_variant_unref);
273
274         g_hash_table_iter_init(&iter, channel_map);
275         while (g_hash_table_iter_next(&iter, &channel_id, &channel_target)) {
276                 if (!channel_target) {
277                         g_printerr("cli: Channel name for \"%s\" missing.\n",
278                                    (char *)channel_id);
279                         continue;
280                 }
281                 ch = find_channel(channel_list, channel_target);
282                 if (!ch) {
283                         g_printerr("cli: No channel with name \"%s\" found.\n",
284                                    (char *)channel_target);
285                         continue;
286                 }
287                 if (!ch->enabled)
288                         g_printerr("cli: Target channel \"%s\" not enabled.\n",
289                                    (char *)channel_target);
290
291                 var = g_variant_new_int32(ch->index);
292                 g_variant_ref_sink(var);
293                 g_hash_table_insert(channel_indices, g_strdup(channel_id), var);
294         }
295
296         srd_inst_channel_set_all(di, channel_indices);
297         g_hash_table_destroy(channel_indices);
298 }
299
300 void map_pd_channels(struct sr_dev_inst *sdi)
301 {
302         GSList *channels;
303
304         channels = sr_dev_inst_channels_get(sdi);
305
306         if (pd_channel_maps) {
307                 g_hash_table_foreach(pd_channel_maps, &map_pd_inst_channels,
308                                      channels);
309                 g_hash_table_destroy(pd_channel_maps);
310                 pd_channel_maps = NULL;
311         }
312 }
313
314 int setup_pd_annotations(char *opt_pd_annotations)
315 {
316         GSList *l, *l_ann;
317         struct srd_decoder *dec;
318         int ann_class;
319         char **pds, **pdtok, **keyval, **annlist, **ann, **ann_descr;
320         const char *dec_id;
321         const char *ann_txt;
322         const char *ann_id;
323         const struct srd_decoder_annotation_row *row_desc;
324         char **ann_diag;
325
326         /* Set up custom list of PDs and annotations to show. */
327         pds = g_strsplit(opt_pd_annotations, ",", 0);
328         for (pdtok = pds; *pdtok && **pdtok; pdtok++) {
329                 keyval = g_strsplit(*pdtok, "=", 0);
330                 dec_id = keyval[0];
331                 if (!(dec = srd_decoder_get_by_id(dec_id))) {
332                         g_critical("Protocol decoder '%s' not found.", dec_id);
333                         g_strfreev(keyval);
334                         g_strfreev(pds);
335                         return 1;
336                 }
337                 if (!dec->annotations) {
338                         g_critical("Protocol decoder '%s' has no annotations.", dec_id);
339                         g_strfreev(keyval);
340                         g_strfreev(pds);
341                         return 1;
342                 }
343                 ann_txt = (g_strv_length(keyval) == 2) ? keyval[1] : NULL;
344                 if (ann_txt && *ann_txt) {
345                         annlist = g_strsplit(ann_txt, ":", 0);
346                         for (ann = annlist; *ann && **ann; ann++) {
347                                 ann_id = *ann;
348                                 g_debug("cli: Lookup decoder %s annotation %s.", dec_id, ann_id);
349                                 /* Lookup annotation class. */
350                                 ann_class = 0;
351                                 for (l = dec->annotations; l; l = l->next, ann_class++) {
352                                         ann_descr = l->data;
353                                         if (!canon_cmp(ann_descr[0], ann_id))
354                                                 /* Found it. */
355                                                 break;
356                                 }
357                                 if (l) {
358                                         l_ann = g_hash_table_lookup(pd_ann_visible, dec_id);
359                                         l_ann = g_slist_append(l_ann, GINT_TO_POINTER(ann_class));
360                                         g_hash_table_replace(pd_ann_visible, g_strdup(dec_id), l_ann);
361                                         g_debug("cli: Showing protocol decoder %s annotation "
362                                                         "class %d (%s).", dec_id, ann_class, ann_descr[0]);
363                                         continue;
364                                 }
365                                 /* Lookup annotation row. */
366                                 for (l = dec->annotation_rows; l; l = l->next) {
367                                         row_desc = l->data;
368                                         if (!canon_cmp(row_desc->id, ann_id))
369                                                 break;
370                                 }
371                                 if (l) {
372                                         g_debug("cli: Showing decoder %s annotation row %s (%s).",
373                                                 dec_id, row_desc->id, row_desc->desc);
374                                         l_ann = g_hash_table_lookup(pd_ann_visible, dec_id);
375                                         for (l = row_desc->ann_classes; l; l = l->next) {
376                                                 /*
377                                                  * This could just be:
378                                                  *   l_ann = g_slist_append(l_ann, l->data);
379                                                  * But we are explicit for readability
380                                                  * and to access details for diagnostics.
381                                                  */
382                                                 ann_class = GPOINTER_TO_INT(l->data);
383                                                 l_ann = g_slist_append(l_ann, GINT_TO_POINTER(ann_class));
384                                                 ann_diag = g_slist_nth_data(dec->annotations, ann_class);
385                                                 g_debug("cli: Adding class %d/%s from row %s.",
386                                                         ann_class, ann_diag[0], row_desc->id);
387                                         }
388                                         g_hash_table_replace(pd_ann_visible, g_strdup(dec_id), l_ann);
389                                         continue;
390                                 }
391                                 /* No match found. */
392                                 g_critical("Annotation '%s' not found "
393                                                 "for protocol decoder '%s'.", ann_id, dec_id);
394                                 g_strfreev(keyval);
395                                 g_strfreev(pds);
396                                 return 1;
397                         }
398                 } else {
399                         /* No class specified: show all of them. */
400                         ann_class = -1;
401                         l_ann = g_slist_append(NULL, GINT_TO_POINTER(ann_class));
402                         g_hash_table_insert(pd_ann_visible, g_strdup(dec_id), l_ann);
403                         g_debug("cli: Showing all annotation classes for protocol "
404                                         "decoder %s.", dec_id);
405                 }
406                 g_strfreev(keyval);
407         }
408         g_strfreev(pds);
409
410         return 0;
411 }
412
413 int setup_pd_meta(char *opt_pd_meta)
414 {
415         struct srd_decoder *dec;
416         char **pds, **pdtok;
417
418         pd_meta_visible = g_hash_table_new_full(g_str_hash, g_int_equal,
419                         g_free, NULL);
420         pds = g_strsplit(opt_pd_meta, ",", 0);
421         for (pdtok = pds; *pdtok && **pdtok; pdtok++) {
422                 if (!(dec = srd_decoder_get_by_id(*pdtok))) {
423                         g_critical("Protocol decoder '%s' not found.", *pdtok);
424                         return 1;
425                 }
426                 g_debug("cli: Showing protocol decoder meta output from '%s'.", *pdtok);
427                 g_hash_table_insert(pd_meta_visible, g_strdup(*pdtok), NULL);
428         }
429         g_strfreev(pds);
430
431         return 0;
432 }
433
434 int setup_pd_binary(char *opt_pd_binary)
435 {
436         GSList *l;
437         struct srd_decoder *dec;
438         int bin_class;
439         char **pds, **pdtok, **keyval, **bin_name;
440
441         pd_binary_visible = g_hash_table_new_full(g_str_hash, g_int_equal,
442                         g_free, NULL);
443         pds = g_strsplit(opt_pd_binary, ",", 0);
444         for (pdtok = pds; *pdtok && **pdtok; pdtok++) {
445                 keyval = g_strsplit(*pdtok, "=", 0);
446                 if (!(dec = srd_decoder_get_by_id(keyval[0]))) {
447                         g_critical("Protocol decoder '%s' not found.", keyval[0]);
448                         return 1;
449                 }
450                 if (!dec->binary) {
451                         g_critical("Protocol decoder '%s' has no binary output.", keyval[0]);
452                         return 1;
453                 }
454                 bin_class = 0;
455                 if (g_strv_length(keyval) == 2) {
456                         for (l = dec->binary; l; l = l->next, bin_class++) {
457                                 bin_name = l->data;
458                                 if (!strcmp(bin_name[0], keyval[1]))
459                                         /* Found it. */
460                                         break;
461                         }
462                         if (!l) {
463                                 g_critical("binary output '%s' not found "
464                                                 "for protocol decoder '%s'.", keyval[1], keyval[0]);
465                                 return 1;
466                         }
467                         g_debug("cli: Showing protocol decoder %s binary class "
468                                         "%d (%s).", keyval[0], bin_class, bin_name[0]);
469                 } else {
470                         /* No class specified: output all of them. */
471                         bin_class = -1;
472                         g_debug("cli: Showing all binary classes for protocol "
473                                         "decoder %s.", keyval[0]);
474                 }
475                 g_hash_table_insert(pd_binary_visible, g_strdup(keyval[0]), GINT_TO_POINTER(bin_class));
476                 g_strfreev(keyval);
477         }
478         g_strfreev(pds);
479
480         return 0;
481 }
482
483 /*
484  * Balance JSON object and array parentheses, and separate array items.
485  * Somewhat convoluted API to re-use the routine for individual items as
486  * well as the surrounding array and object, including deferred start of
487  * the output and late flush (and to keep the state strictly local to the
488  * routine). Some additional complexity due to JSON's inability to handle
489  * a trailing comma at the last item. Code phrased such that text literals
490  * are kept in their order of appearance in the output (where possible).
491  */
492 static void jsontrace_open_close(gboolean is_close_req,
493         gboolean open_item, gboolean close_item)
494 {
495         static gboolean is_file_open;
496         static gboolean is_item_open;
497
498         if (is_close_req && is_item_open)
499                 close_item = TRUE;
500
501         /* Automatic file header, and array item separation. */
502         if (open_item) {
503                 if (!is_file_open)
504                         printf("{\"traceEvents\": [\n");
505                 if (is_item_open) {
506                         printf("}");
507                         is_item_open = FALSE;
508                 }
509                 if (is_file_open) {
510                         printf(",\n");
511                 }
512                 is_file_open = TRUE;
513         }
514
515         /* Array item open/append/close. */
516         if (open_item) {
517                 printf("{");
518                 is_item_open = TRUE;
519         }
520         if (!open_item && !close_item && !is_close_req) {
521                 printf(", ");
522                 is_item_open = TRUE;
523         }
524         if (close_item) {
525                 printf("}");
526                 is_item_open = FALSE;
527         }
528
529         /* Automatic file footer on shutdown. */
530         if (is_close_req && is_file_open) {
531                 printf("\n");
532                 printf("]}\n");
533         }
534         if (is_close_req)
535                 is_file_open = FALSE;
536
537         /* Flush at end of lines, or end of file. */
538         if (close_item || is_close_req)
539                 fflush(stdout);
540 }
541
542 /* Convert uint64 sample number to double timestamp in microseconds. */
543 static double jsontrace_ts_usec(uint64_t snum)
544 {
545         double ts_usec;
546
547         ts_usec = snum;
548         ts_usec *= 1e6;
549         ts_usec /= pd_samplerate;
550         return ts_usec;
551 }
552
553 /* Emit two Google Trace Events (JSON) for one PD annotation (ss, es). */
554 static void jsontrace_annotation(struct srd_decoder *dec,
555         struct srd_proto_data_annotation *pda, struct srd_proto_data *pdata)
556 {
557         char *row_text;
558         GSList *lrow, *lcls;
559         struct srd_decoder_annotation_row *row;
560         int cls;
561         char **ann_descr;
562
563         /*
564          * Search for an annotation row for this index, or use the
565          * annotation's descriptor.
566          */
567         row_text = NULL;
568         if (dec->annotation_rows) {
569                 for (lrow = dec->annotation_rows; lrow; lrow = lrow->next) {
570                         row = lrow->data;
571                         for (lcls = row->ann_classes; lcls; lcls = lcls->next) {
572                                 cls = GPOINTER_TO_INT(lcls->data);
573                                 if (cls == pda->ann_class) {
574                                         row_text = row->desc;
575                                         break;
576                                 }
577                         }
578                         if (row_text)
579                                 break;
580                 }
581         }
582         if (!row_text) {
583                 ann_descr = g_slist_nth_data(dec->annotations, pda->ann_class);
584                 row_text = ann_descr[0];
585         }
586
587         /*
588          * Emit two Google Trace Events for the start and end times.
589          * Set the 'pid' (process ID) to the decoder name to group a
590          * decoder's annotations. Set the 'tid' (thread ID) to the
591          * annotation row's description. The 'ts' (timestamp) is in
592          * microseconds. Set 'name' to the longest annotation text.
593          *
594          * BEWARE of the unfortunate JSON format limitation, which
595          * clutters data output calls with format helper calls.
596          * TODO Want to introduce a cJSON dependency to delegate the
597          * construction of output text?
598          */
599         jsontrace_open_close(FALSE, TRUE, FALSE);
600         printf("\"%s\": \"%s\"", "ph", "B");
601         jsontrace_open_close(FALSE, FALSE, FALSE);
602         printf("\"%s\": %lf", "ts", jsontrace_ts_usec(pdata->start_sample));
603         jsontrace_open_close(FALSE, FALSE, FALSE);
604         printf("\"%s\": \"%s\"", "pid", pdata->pdo->proto_id);
605         jsontrace_open_close(FALSE, FALSE, FALSE);
606         printf("\"%s\": \"%s\"", "tid", row_text);
607         jsontrace_open_close(FALSE, FALSE, FALSE);
608         printf("\"%s\": \"%s\"", "name", pda->ann_text[0]);
609
610         jsontrace_open_close(FALSE, TRUE, FALSE);
611         printf("\"%s\": \"%s\"", "ph", "E");
612         jsontrace_open_close(FALSE, FALSE, FALSE);
613         printf("\"%s\": %lf", "ts", jsontrace_ts_usec(pdata->end_sample));
614         jsontrace_open_close(FALSE, FALSE, FALSE);
615         printf("\"%s\": \"%s\"", "pid", pdata->pdo->proto_id);
616         jsontrace_open_close(FALSE, FALSE, FALSE);
617         printf("\"%s\": \"%s\"", "tid", row_text);
618         jsontrace_open_close(FALSE, FALSE, FALSE);
619         printf("\"%s\": \"%s\"", "name", pda->ann_text[0]);
620
621         jsontrace_open_close(FALSE, FALSE, TRUE);
622 }
623
624 void show_pd_annotations(struct srd_proto_data *pdata, void *cb_data)
625 {
626         struct srd_decoder *dec;
627         struct srd_proto_data_annotation *pda;
628         GSList *ann_list, *l;
629         int i;
630         char **ann_descr;
631         gboolean show_ann, show_snum, show_class, show_quotes, show_abbrev;
632         const char *quote;
633
634         (void)cb_data;
635
636         if (!pd_ann_visible)
637                 return;
638
639         if (!g_hash_table_lookup_extended(pd_ann_visible, pdata->pdo->di->decoder->id,
640                         NULL, (void **)&ann_list)) {
641                 /* Not in the list of PDs whose annotations we're showing. */
642                 return;
643         }
644
645         dec = pdata->pdo->di->decoder;
646         pda = pdata->data;
647         show_ann = FALSE;
648         for (l = ann_list; l; l = l->next) {
649                 if (GPOINTER_TO_INT(l->data) == -1
650                                 || GPOINTER_TO_INT(l->data) == pda->ann_class) {
651                         show_ann = TRUE;
652                         break;
653                 }
654         }
655         if (!show_ann)
656                 return;
657
658         /* Google Trace Events are rather special. Use a separate code path. */
659         if (opt_pd_jsontrace) {
660                 jsontrace_annotation(dec, pda, pdata);
661                 return;
662         }
663
664         /*
665          * Determine which fields of the annotation to display. Inspect
666          * user specified options as well as the verbosity of the log level:
667          * - Optionally show the sample numbers for the annotation's span.
668          * - Always show the protocol decoder ID.
669          * - Optionally show the annotation's class description.
670          * - Always show the longest annotation text.
671          * - Optionally show alternative annotation text (abbreviations
672          *   for different zoom levels).
673          * - Optionally put quote marks around annotation text, when
674          *   recipients might have to deal with a set of text variants.
675          */
676         show_snum = show_class = show_quotes = show_abbrev = FALSE;
677         if (opt_pd_samplenum || opt_loglevel > SR_LOG_WARN) {
678                 show_snum = TRUE;
679         }
680         if (opt_loglevel > SR_LOG_WARN) {
681                 show_quotes = TRUE;
682         }
683         if (opt_loglevel > SR_LOG_INFO) {
684                 show_class = TRUE;
685                 show_abbrev = TRUE;
686         }
687
688         /*
689          * Display the annotation's fields after the layout was
690          * determined above.
691          */
692         if (show_snum) {
693                 printf("%" PRIu64 "-%" PRIu64 " ",
694                         pdata->start_sample, pdata->end_sample);
695         }
696         printf("%s: ", pdata->pdo->proto_id);
697         if (show_class) {
698                 ann_descr = g_slist_nth_data(dec->annotations, pda->ann_class);
699                 printf("%s: ", ann_descr[0]);
700         }
701         quote = show_quotes ? "\"" : "";
702         printf("%s%s%s", quote, pda->ann_text[0], quote);
703         if (show_abbrev) {
704                 for (i = 1; pda->ann_text[i]; i++)
705                         printf(" %s%s%s", quote, pda->ann_text[i], quote);
706         }
707         printf("\n");
708         fflush(stdout);
709 }
710
711 void show_pd_meta(struct srd_proto_data *pdata, void *cb_data)
712 {
713         (void)cb_data;
714
715         if (!g_hash_table_lookup_extended(pd_meta_visible,
716                         pdata->pdo->di->decoder->id, NULL, NULL))
717                 /* Not in the list of PDs whose meta output we're showing. */
718                 return;
719
720         if (opt_pd_samplenum || opt_loglevel > SR_LOG_WARN)
721                 printf("%"PRIu64"-%"PRIu64" ", pdata->start_sample, pdata->end_sample);
722         printf("%s: ", pdata->pdo->proto_id);
723         printf("%s: %s", pdata->pdo->meta_name, g_variant_print(pdata->data, FALSE));
724         printf("\n");
725         fflush(stdout);
726 }
727
728 void show_pd_binary(struct srd_proto_data *pdata, void *cb_data)
729 {
730         struct srd_proto_data_binary *pdb;
731         gpointer classp;
732         int classi;
733
734         (void)cb_data;
735
736         if (!g_hash_table_lookup_extended(pd_binary_visible,
737                         pdata->pdo->di->decoder->id, NULL, (void **)&classp))
738                 /* Not in the list of PDs whose meta output we're showing. */
739                 return;
740
741         classi = GPOINTER_TO_INT(classp);
742         pdb = pdata->data;
743         if (classi != -1 && classi != pdb->bin_class)
744                 /* Not showing this binary class. */
745                 return;
746
747         /* Just send the binary output to stdout, no embellishments. */
748         fwrite(pdb->data, pdb->size, 1, stdout);
749         fflush(stdout);
750 }
751
752 void show_pd_prepare(void)
753 {
754         if (opt_pd_jsontrace)
755                 jsontrace_open_close(TRUE, FALSE, FALSE);
756 }
757
758 void show_pd_close(void)
759 {
760         if (opt_pd_jsontrace)
761                 jsontrace_open_close(TRUE, FALSE, FALSE);
762 }
763 #endif