summaryrefslogtreecommitdiff
path: root/core/upb_msg.c
blob: 75f7a35fb4212f56f942ea3df1334b5b48a59c80 (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
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
/*
 * upb - a minimalist implementation of protocol buffers.
 *
 * Copyright (c) 2010 Joshua Haberman.  See LICENSE for details.
 *
 * Data structure for storing a message of protobuf data.
 */

#include "upb_msg.h"

void _upb_elem_free(upb_value v, upb_fielddef *f) {
  switch(f->type) {
    case UPB_TYPE(MESSAGE):
    case UPB_TYPE(GROUP):
      _upb_msg_free(v.msg, upb_downcast_msgdef(f->def));
      break;
    case UPB_TYPE(STRING):
    case UPB_TYPE(BYTES):
      _upb_string_free(v.str);
      break;
    default:
      abort();
  }
}

void _upb_field_free(upb_value v, upb_fielddef *f) {
  if (upb_isarray(f)) {
    _upb_array_free(v.arr, f);
  } else {
    _upb_elem_free(v, f);
  }
}

upb_msg *upb_msg_new(upb_msgdef *md) {
  upb_msg *msg = malloc(md->size);
  // Clear all set bits and cached pointers.
  memset(msg, 0, md->size);
  upb_atomic_refcount_init(&msg->refcount, 1);
  return msg;
}

void _upb_msg_free(upb_msg *msg, upb_msgdef *md) {
  // Need to release refs on all sub-objects.
  upb_msg_iter i;
  for(i = upb_msg_begin(md); !upb_msg_done(i); i = upb_msg_next(md, i)) {
    upb_fielddef *f = upb_msg_iter_field(i);
    upb_valueptr p = _upb_msg_getptr(msg, f);
    upb_valuetype_t type = upb_field_valuetype(f);
    if (upb_field_ismm(f)) _upb_field_unref(upb_value_read(p, type), f);
  }
  free(msg);
}

upb_array *upb_array_new(void) {
  upb_array *arr = malloc(sizeof(*arr));
  upb_atomic_refcount_init(&arr->refcount, 1);
  arr->size = 0;
  arr->len = 0;
  arr->elements._void = NULL;
  return arr;
}

void _upb_array_free(upb_array *arr, upb_fielddef *f) {
  if (upb_elem_ismm(f)) {
    // Need to release refs on sub-objects.
    upb_valuetype_t type = upb_elem_valuetype(f);
    for (upb_arraylen_t i = 0; i < arr->size; i++) {
      upb_valueptr p = _upb_array_getptr(arr, f, i);
      _upb_elem_unref(upb_value_read(p, type), f);
    }
  }
  if (arr->elements._void) free(arr->elements._void);
  free(arr);
}

upb_value upb_field_new(upb_fielddef *f, upb_valuetype_t type) {
  upb_value v;
  switch(type) {
    case UPB_TYPE(MESSAGE):
    case UPB_TYPE(GROUP):
      v.msg = upb_msg_new(upb_downcast_msgdef(f->def));
    case UPB_TYPE(STRING):
    case UPB_TYPE(BYTES):
      v.str = upb_string_new();
    case UPB_VALUETYPE_ARRAY:
      v.arr = upb_array_new();
    default:
      abort();
  }
  return v;
}

static void upb_field_recycle(upb_value val) {
  (void)val;
}

upb_value upb_field_tryrecycle(upb_valueptr p, upb_value val, upb_fielddef *f,
                               upb_valuetype_t type) {
  if (val._void == NULL || !upb_atomic_only(val.refcount)) {
    if (val._void != NULL) upb_atomic_unref(val.refcount);
    val = upb_field_new(f, type);
    upb_value_write(p, val, type);
  } else {
    upb_field_recycle(val);
  }
  return val;
}

void upb_msg_decodestr(upb_msg *msg, upb_msgdef *md, upb_string *str,
                       upb_status *status) {
  (void)msg;
  (void)md;
  (void)str;
  (void)status;
}

void upb_msg_encodestr(upb_msg *msg, upb_msgdef *md, upb_string *str,
                       upb_status *status) {
  (void)msg;
  (void)md;
  (void)str;
  (void)status;
}
generated by cgit on debian on lair
contact matthew@masot.net with questions or feedback