+CYType *Pointer::Decode(CYPool &pool) const {
+ return CYDecodeType(pool, &type)->Modify($ CYTypePointerTo());
+}
+
+CYType *Array::Decode(CYPool &pool) const {
+ return CYDecodeType(pool, &type)->Modify($ CYTypeArrayOf($D(size)));
+}
+
+#ifdef CY_OBJECTIVEC
+CYType *Object::Decode(CYPool &pool) const {
+ if (name == NULL)
+ return $ CYType($ CYTypeVariable("id"));
+ else
+ return $ CYType($ CYTypeVariable(name), $ CYTypePointerTo());
+}
+#endif
+
+CYType *Enum::Decode(CYPool &pool) const {
+ CYEnumConstant *values(NULL);
+ for (size_t i(count); i != 0; --i)
+ values = $ CYEnumConstant($I(pool.strdup(constants[i - 1].name)), $D(constants[i - 1].value), values);
+ CYIdentifier *identifier(name == NULL ? NULL : $I(name));
+ CYType *typed(type.Decode(pool));
+ _assert(typed->modifier_ == NULL);
+ return $ CYType($ CYTypeEnum(identifier, typed->specifier_, values));
+}
+
+CYType *Aggregate::Decode(CYPool &pool) const {
+ _assert(!overlap);
+
+ if (signature.count == _not(size_t)) {
+ _assert(name != NULL);
+ return $ CYType($ CYTypeReference(CYTypeReferenceStruct, $I($pool.strdup(name))));
+ }
+
+ CYTypeStructField *fields(NULL);
+ for (size_t i(signature.count); i != 0; --i) {
+ sig::Element &element(signature.elements[i - 1]);
+ fields = $ CYTypeStructField(CYDecodeType(pool, element.type), element.name == NULL ? NULL : $I(element.name), fields);
+ }
+ CYIdentifier *identifier(name == NULL ? NULL : $I(name));
+ return $ CYType($ CYTypeStruct(identifier, $ CYStructTail(fields)));
+}
+
+CYType *Callable::Decode(CYPool &pool) const {
+ _assert(signature.count != 0);
+ CYTypedParameter *parameters(NULL);
+ for (size_t i(signature.count - 1); i != 0; --i)
+ parameters = $ CYTypedParameter(CYDecodeType(pool, signature.elements[i].type), NULL, parameters);
+ return Modify(pool, CYDecodeType(pool, signature.elements[0].type), parameters);
+}
+
+CYType *Function::Modify(CYPool &pool, CYType *result, CYTypedParameter *parameters) const {
+ return result->Modify($ CYTypeFunctionWith(variadic, parameters));
+}
+
+#ifdef CY_OBJECTIVEC
+CYType *Block::Modify(CYPool &pool, CYType *result, CYTypedParameter *parameters) const {
+ return result->Modify($ CYTypeBlockWith(parameters));
+}
+
+CYType *Block::Decode(CYPool &pool) const {
+ if (signature.count == 0)
+ return $ CYType($ CYTypeVariable("NSBlock"), $ CYTypePointerTo());
+ return Callable::Decode(pool);
+}
+#endif
+
+}
+
+CYType *CYDecodeType(CYPool &pool, struct sig::Type *type) {
+ CYType *typed(type->Decode(pool));
+ if ((type->flags & JOC_TYPE_CONST) != 0) {
+ if (dynamic_cast<sig::String *>(type) != NULL)
+ typed->modifier_ = $ CYTypeConstant(typed->modifier_);
+ else
+ typed = typed->Modify($ CYTypeConstant());
+ }