Finish upb_context thread-safety.

pull/13171/head
Joshua Haberman 16 years ago
parent 71d82d06d1
commit 7fa19a3f95
  1. 44
      src/upb_context.c
  2. 2
      src/upb_context.h
  3. 6
      tests/tests.c
  4. 13
      tools/upbc.c

@ -23,8 +23,9 @@ bool addfd(struct upb_strtable *addto, struct upb_strtable *existingdefs,
google_protobuf_FileDescriptorProto *fd, bool sort, google_protobuf_FileDescriptorProto *fd, bool sort,
struct upb_context *context); struct upb_context *context);
bool upb_context_init(struct upb_context *c) struct upb_context *upb_context_new()
{ {
struct upb_context *c = malloc(sizeof(*c));
upb_atomic_refcount_init(&c->refcount, 1); upb_atomic_refcount_init(&c->refcount, 1);
upb_rwlock_init(&c->lock); upb_rwlock_init(&c->lock);
upb_strtable_init(&c->symtab, 16, sizeof(struct upb_symtab_entry)); upb_strtable_init(&c->symtab, 16, sizeof(struct upb_symtab_entry));
@ -34,7 +35,7 @@ bool upb_context_init(struct upb_context *c)
upb_file_descriptor_set->file->elements[0]; /* We know there is only 1. */ upb_file_descriptor_set->file->elements[0]; /* We know there is only 1. */
if(!addfd(&c->psymtab, &c->symtab, fd, false, c)) { if(!addfd(&c->psymtab, &c->symtab, fd, false, c)) {
assert(false); assert(false);
return false; /* Indicates that upb is buggy or corrupt. */ return NULL; /* Indicates that upb is buggy or corrupt. */
} }
struct upb_string name = UPB_STRLIT("google.protobuf.FileDescriptorSet"); struct upb_string name = UPB_STRLIT("google.protobuf.FileDescriptorSet");
struct upb_symtab_entry *e = upb_strtable_lookup(&c->psymtab, &name); struct upb_symtab_entry *e = upb_strtable_lookup(&c->psymtab, &name);
@ -43,7 +44,7 @@ bool upb_context_init(struct upb_context *c)
c->fds_size = 16; c->fds_size = 16;
c->fds_len = 0; c->fds_len = 0;
c->fds = malloc(sizeof(*c->fds)); c->fds = malloc(sizeof(*c->fds));
return true; return c;
} }
static void free_symtab(struct upb_strtable *t) static void free_symtab(struct upb_strtable *t)
@ -68,6 +69,7 @@ static void free_context(struct upb_context *c)
upb_msg_free((struct upb_msg*)c->fds[i]); upb_msg_free((struct upb_msg*)c->fds[i]);
free_symtab(&c->psymtab); free_symtab(&c->psymtab);
free(c->fds); free(c->fds);
free(c);
} }
void upb_context_unref(struct upb_context *c) void upb_context_unref(struct upb_context *c)
@ -83,7 +85,10 @@ void upb_context_unref(struct upb_context *c)
struct upb_symtab_entry *upb_context_lookup(struct upb_context *c, struct upb_symtab_entry *upb_context_lookup(struct upb_context *c,
struct upb_string *symbol) struct upb_string *symbol)
{ {
return upb_strtable_lookup(&c->symtab, symbol); upb_rwlock_rdlock(&c->lock);
struct upb_symtab_entry *e = upb_strtable_lookup(&c->symtab, symbol);
upb_rwlock_unlock(&c->lock);
return e;
} }
/* Given a symbol and the base symbol inside which it is defined, find the /* Given a symbol and the base symbol inside which it is defined, find the
@ -137,7 +142,10 @@ union upb_symbol_ref resolve2(struct upb_strtable *t1, struct upb_strtable *t2,
struct upb_symtab_entry *upb_context_resolve(struct upb_context *c, struct upb_symtab_entry *upb_context_resolve(struct upb_context *c,
struct upb_string *base, struct upb_string *base,
struct upb_string *symbol) { struct upb_string *symbol) {
return resolve(&c->symtab, base, symbol); upb_rwlock_rdlock(&c->lock);
struct upb_symtab_entry *e = resolve(&c->symtab, base, symbol);
upb_rwlock_unlock(&c->lock);
return e;
} }
/* Joins strings together, for example: /* Joins strings together, for example:
@ -280,17 +288,25 @@ bool upb_context_addfds(struct upb_context *c,
* the descriptor is valid. */ * the descriptor is valid. */
struct upb_strtable tmp; struct upb_strtable tmp;
upb_strtable_init(&tmp, 0, sizeof(struct upb_symtab_entry)); upb_strtable_init(&tmp, 0, sizeof(struct upb_symtab_entry));
upb_rwlock_rdlock(&c->lock);
for(uint32_t i = 0; i < fds->file->len; i++) { for(uint32_t i = 0; i < fds->file->len; i++) {
if(!addfd(&tmp, &c->symtab, fds->file->elements[i], true, c)) { if(!addfd(&tmp, &c->symtab, fds->file->elements[i], true, c)) {
printf("Not added successfully!\n"); printf("Not added successfully!\n");
free_symtab(&tmp); free_symtab(&tmp);
upb_rwlock_unlock(&c->lock);
return false; return false;
} }
} }
upb_rwlock_unlock(&c->lock);
/* Everything was successfully added, copy from the tmp symtable. */ /* Everything was successfully added, copy from the tmp symtable. */
struct upb_symtab_entry *e; struct upb_symtab_entry *e;
for(e = upb_strtable_begin(&tmp); e; e = upb_strtable_next(&tmp, &e->e)) {
upb_strtable_insert(&c->symtab, &e->e); upb_rwlock_wrlock(&c->lock);
for(e = upb_strtable_begin(&tmp); e; e = upb_strtable_next(&tmp, &e->e))
upb_strtable_insert(&c->symtab, &e->e);
upb_rwlock_unlock(&c->lock);
}
upb_strtable_free(&tmp); upb_strtable_free(&tmp);
} }
return true; return true;
@ -302,11 +318,15 @@ bool upb_context_parsefds(struct upb_context *c, struct upb_string *fds_str) {
if(!fds) return false; if(!fds) return false;
if(!upb_context_addfds(c, fds)) return false; if(!upb_context_addfds(c, fds)) return false;
/* We own fds now, need to keep a ref so we can free it later. */ {
if(c->fds_size == c->fds_len) { /* We own fds now, need to keep a ref so we can free it later. */
c->fds_size *= 2; upb_rwlock_wrlock(&c->lock);
c->fds = realloc(c->fds, c->fds_size); if(c->fds_size == c->fds_len) {
c->fds_size *= 2;
c->fds = realloc(c->fds, c->fds_size);
}
c->fds[c->fds_len++] = fds;
upb_rwlock_unlock(&c->lock);
} }
c->fds[c->fds_len++] = fds;
return true; return true;
} }

@ -53,7 +53,7 @@ struct upb_context {
/* Initializes a upb_context. Contexts are not freed explicitly, but unref'd /* Initializes a upb_context. Contexts are not freed explicitly, but unref'd
* when the caller is done with them. */ * when the caller is done with them. */
bool upb_context_init(struct upb_context *c); struct upb_context *upb_context_new();
INLINE void upb_context_ref(struct upb_context *c) { INLINE void upb_context_ref(struct upb_context *c) {
upb_atomic_ref(&c->refcount); upb_atomic_ref(&c->refcount);
} }

@ -222,9 +222,9 @@ static void test_get_f_uint32_t()
} }
static void test_upb_context() { static void test_upb_context() {
struct upb_context c; struct upb_context *c = upb_context_new();
ASSERT(upb_context_init(&c)); ASSERT(c);
upb_context_unref(&c); upb_context_unref(c);
} }
int main() int main()

@ -646,13 +646,12 @@ int main(int argc, char *argv[])
error("Couldn't read input file."); error("Couldn't read input file.");
/* Parse input file. */ /* Parse input file. */
struct upb_context c; struct upb_context *c = upb_context_new();
upb_context_init(&c); struct upb_msg *fds_msg = upb_msg_parsenew(c->fds_msg, descriptor);
struct upb_msg *fds_msg = upb_msg_parsenew(c.fds_msg, descriptor);
google_protobuf_FileDescriptorSet *fds = (void*)fds_msg; google_protobuf_FileDescriptorSet *fds = (void*)fds_msg;
if(!fds) if(!fds)
error("Failed to parse input file descriptor."); error("Failed to parse input file descriptor.");
if(!upb_context_addfds(&c, fds)) if(!upb_context_addfds(c, fds))
error("Failed to resolve symbols in descriptor.\n"); error("Failed to resolve symbols in descriptor.\n");
/* We need to sort the fields of all the descriptors. They will already be /* We need to sort the fields of all the descriptors. They will already be
@ -676,17 +675,17 @@ int main(int argc, char *argv[])
if(!h_file) error("Failed to open .h output file"); if(!h_file) error("Failed to open .h output file");
int symcount; int symcount;
struct upb_symtab_entry **entries = strtable_to_array(&c.symtab, &symcount); struct upb_symtab_entry **entries = strtable_to_array(&c->symtab, &symcount);
write_h(entries, symcount, h_filename, cident, h_file); write_h(entries, symcount, h_filename, cident, h_file);
free(entries); free(entries);
if(cident) { if(cident) {
FILE *c_file = fopen(c_filename, "w"); FILE *c_file = fopen(c_filename, "w");
if(!c_file) error("Failed to open .h output file"); if(!c_file) error("Failed to open .h output file");
write_message_c(fds, c.fds_msg, cident, h_filename, argc, argv, input_file, c_file); write_message_c(fds, c->fds_msg, cident, h_filename, argc, argv, input_file, c_file);
fclose(c_file); fclose(c_file);
} }
upb_msg_free(fds_msg); upb_msg_free(fds_msg);
upb_context_unref(&c); upb_context_unref(c);
upb_strfree(descriptor); upb_strfree(descriptor);
fclose(h_file); fclose(h_file);

Loading…
Cancel
Save