Fleshed out DESIGN.md a bit more.
diff --git a/DESIGN.md b/DESIGN.md
index a1f0eed..73388d0 100644
--- a/DESIGN.md
+++ b/DESIGN.md
@@ -160,43 +160,42 @@
together, their lifetimes are irreversibly joined, such that none of the arena
blocks in either arena will be freed until *both* arenas are freed with
`upb_arena_free()`. This is useful when joining two messages from separate
-arenas, making one a sub-message of the other. Fuse is an a very cheap
+arenas (making one a sub-message of the other). Fuse is an a very cheap
operation, and an unlimited number of arenas can be fused together efficiently.
-## Binary Parsing and Serialzation
+## Reflection and Descriptors
-For binary format parsing and serializing, we use tables of fields known as
-*mini-tables*. (The "mini" distinguishes them from "fast tables", which are
-a larger and more optimized table format used by the fast parser in
-`upb/decode_fast.c`.)
+upb offers a fully-featured reflection library. There are two main ways of
+using reflection:
-The format of mini-tables is defined in `upb/msg_internal.h`. As the name
-suggests, the format of these mini-tables is internal-only, consumed by the
-parser and serializer, but not available for general use by users. The format
-of these tables is strongly aimed at making the parser and serializer as fast
-as possible, and this sometimes involves changing them in backward-incompatible
-ways.
+1. You can load descriptors from strings using `upb_symtab_addfile()`.
+ The upb runtime will dynamically create mini-tables like what the upb compiler
+ would have created if you had compiled this type into a `.upb.c` file.
+2. You can load descriptors using generated `.upbdefs.h` interfaces.
+ This will load reflection that references the corresponding `.upb.c`
+ mini-tables instead of building a new mini-table on the fly. This lets
+ you reflect on generated types that are linked into your program.
-These tables define field numbers, field types, and offsets for every field.
-It is important that these offsets match the offsets used in the generated
-accessors, for obvious reasons.
+upb's design for descriptors is similar to protobuf C++ in many ways, with
+the following correspondences:
-The generated `.upb.h` interface exposes wrappers for parsing and serialization
-that automatically pass the appropriate mini-tables to the parser and serializer:
+| C++ Type | upb type |
+| ---------| ---------|
+| `google::protobuf::DescriptorPool` | `upb_symtab`
+| `google::protobuf::Descriptor` | `upb_msgdef`
+| `google::protobuf::FieldDescriptor` | `upb_fielddef`
+| `google::protobuf::OneofDescriptor` | `upb_oneofdef`
+| `google::protobuf::EnumDescriptor` | `upb_enumdef`
+| `google::protobuf::FileDescriptor` | `upb_filedef`
+| `google::protobuf::ServiceDescriptor` | `upb_servicedef`
+| `google::protobuf::MethodDescriptor` | `upb_methoddef`
-```c
-#include "google/protobuf/descriptor.upb.h"
+Like in C++ descriptors (defs) are created by loading a
+`google_protobuf_FileDescriptorProto` into a `upb_symtab`. This creates and
+links all of the def objects corresponding to that `.proto` file, and inserts
+the names into a symbol table so they can be looked up by name.
-bool ParseDescriptor(const char *pb_data, size_t pb_size) {
- // Arena where all messages, arrays, maps, etc. will be allocated.
- upb_arena *arena = upb_arena_new();
-
- // This will pass the mini-table to upb_decode().
- google_protobuf_DescriptorProto* descriptor =
- google_protobuf_DescriptorProto_parse(pb_data, pb_size, arena);
-
- bool ok = descriptor != NULL;
- upb_arena_free(arena);
- return ok;
-}
-```
+Once you have loaded some descriptors into a `upb_symtab`, you can create and
+manipulate messages using the interfaces defined in `upb/reflection.h`. If your
+descriptors are linked to your generated layouts using option (2) above, you can
+safely access the same messages using both reflection and generated interfaces.