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