summaryrefslogtreecommitdiff
path: root/benchmarks/parsestream.upb.c
blob: 0d709a494931537ed260ed6eb57ec26c296e9345 (plain)
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

#include "main.c"

#include <stdlib.h>
#include "upb/bytestream.h"
#include "upb/def.h"
#include "upb/pb/decoder.h"
#include "upb/pb/glue.h"

static char *input_str;
static size_t input_len;
static const upb_msgdef *def;
upb_pipeline pipeline;
static upb_sink *sink;

static void *startsubmsg(const upb_sinkframe *frame) {
  UPB_UNUSED(frame);
  return input_str;
}

void onmreg(void *c, upb_handlers *h) {
  upb_msg_iter i;
  upb_msg_begin(&i, upb_handlers_msgdef(h));
  for(; !upb_msg_done(&i); upb_msg_next(&i)) {
    const upb_fielddef *f = upb_msg_iter_field(&i);
    if (upb_fielddef_type(f) == UPB_TYPE_MESSAGE) {
      upb_handlers_setstartsubmsg(h, f, startsubmsg, NULL, NULL);
    }
  }
  UPB_UNUSED(c);
}

static bool initialize()
{
  // Initialize upb state, decode descriptor.
  upb_status status = UPB_STATUS_INIT;
  upb_symtab *s = upb_symtab_new(&s);
  upb_load_descriptor_file_into_symtab(s, MESSAGE_DESCRIPTOR_FILE, &status);
  if(!upb_ok(&status)) {
    fprintf(stderr, "Error reading descriptor: %s\n",
            upb_status_getstr(&status));
    return false;
  }

  def = upb_dyncast_msgdef(upb_symtab_lookup(s, MESSAGE_NAME, &def));
  if(!def) {
    fprintf(stderr, "Error finding symbol '%s'.\n", MESSAGE_NAME);
    return false;
  }
  upb_symtab_unref(s, &s);

  // Read the message data itself.
  input_str = upb_readfile(MESSAGE_FILE, &input_len);
  if(input_str == NULL) {
    fprintf(stderr, "Error reading " MESSAGE_FILE "\n");
    return false;
  }

  // Cause all messages to be read, but do nothing when they are.
  const upb_handlers* handlers =
      upb_handlers_newfrozen(def, NULL, &handlers, &onmreg, NULL);
  const upb_handlers* decoder_handlers =
      upb_pbdecoder_gethandlers(handlers, JIT, &decoder_handlers);
  upb_msgdef_unref(def, &def);

  upb_pipeline_init(&pipeline, NULL, 0, upb_realloc, NULL);
  upb_sink *s2 = upb_pipeline_newsink(&pipeline, handlers);
  sink = upb_pipeline_newsink(&pipeline, decoder_handlers);
  upb_pipeline_donateref(&pipeline, decoder_handlers, &decoder_handlers);
  upb_pipeline_donateref(&pipeline, handlers, &handlers);
  upb_pbdecoder *decoder = upb_sinkframe_userdata(upb_sink_top(sink));
  upb_pbdecoder_resetsink(decoder, s2);
  return true;
}

static void cleanup()
{
  free(input_str);
  upb_pipeline_uninit(&pipeline);
}

static size_t run(int i)
{
  (void)i;
  upb_pipeline_reset(&pipeline);
  if (!upb_bytestream_putstr(sink, input_str, input_len)) {
    fprintf(stderr, "Decode error: %s", upb_status_getstr(upb_pipeline_status(&pipeline)));
    return 0;
  }
  return input_len;

}
generated by cgit on debian on lair
contact matthew@masot.net with questions or feedback