X-Git-Url: https://git.saurik.com/cycript.git/blobdiff_plain/2fad14e52c8cde8c45003a2ebb6907a57ca380e4..1fe330114b9b59b11dea95b8f941f478eae8f020:/Replace.cpp diff --git a/Replace.cpp b/Replace.cpp index ed486f7..d499dbc 100644 --- a/Replace.cpp +++ b/Replace.cpp @@ -1,5 +1,5 @@ -/* Cycript - Optimizing JavaScript Compiler/Runtime - * Copyright (C) 2009-2015 Jay Freeman (saurik) +/* Cycript - The Truly Universal Scripting Language + * Copyright (C) 2009-2016 Jay Freeman (saurik) */ /* GNU Affero General Public License, Version 3 {{{ */ @@ -117,11 +117,22 @@ CYTarget *CYArrayComprehension::Replace(CYContext &context) { } CYExpression *CYAssignment::Replace(CYContext &context) { + // XXX: this is a horrible hack but I'm a month over schedule :( + if (CYSubscriptMember *subscript = dynamic_cast(lhs_)) + return $C2($M(subscript->object_, $S("$cys")), subscript->property_, rhs_); context.Replace(lhs_); context.Replace(rhs_); return this; } +CYTarget *CYAttemptMember::Replace(CYContext &context) { + CYIdentifier *value(context.Unique()); + + return $C1($F(NULL, $P1($B(value)), $$ + ->* $ CYReturn($ CYCondition($V(value), $M($V(value), property_), $V(value))) + ), object_); +} + CYStatement *CYBlock::Return() { CYImplicitReturn(code_); return this; @@ -142,6 +153,15 @@ CYStatement *CYBreak::Replace(CYContext &context) { } CYTarget *CYCall::Replace(CYContext &context) { + // XXX: this also is a horrible hack but I'm still a month over schedule :( + if (CYAttemptMember *member = dynamic_cast(function_)) { + CYIdentifier *value(context.Unique()); + + return $C1($F(NULL, $P1($B(value)), $$ + ->* $ CYReturn($ CYCondition($V(value), $C($M($V(value), member->property_), arguments_), $V(value))) + ), member->object_); + } + context.Replace(function_); arguments_->Replace(context); return this; @@ -179,6 +199,7 @@ CYTarget *CYClassExpression::Replace(CYContext &context) { if (tail_->constructor_ == NULL) tail_->constructor_ = $ CYFunctionExpression(NULL, NULL, NULL); + tail_->constructor_->name_ = name_; tail_->constructor_ = CYSuperize(context, tail_->constructor_); context.super_ = old; @@ -381,8 +402,19 @@ CYFunctionParameter *CYExpression::Parameter() const { return NULL; } -CYStatement *CYExternal::Replace(CYContext &context) { - return $E($ CYAssign($V(typed_->identifier_), $C1(typed_->Replace(context), $C2($V("dlsym"), $V("RTLD_DEFAULT"), $S(typed_->identifier_->Word()))))); +CYTarget *CYExtend::Replace(CYContext &context) { + return object_.Replace(context, lhs_); +} + +CYStatement *CYExternalDefinition::Replace(CYContext &context) { + return $E($ CYAssign($V(name_), $ CYExternalExpression(abi_, type_, name_))); +} + +CYTarget *CYExternalExpression::Replace(CYContext &context) { + CYExpression *expression(name_->Number(context)); + if (expression == NULL) + expression = $C2($V("dlsym"), $V("RTLD_DEFAULT"), name_->PropertyName(context)); + return $C1(type_->Replace(context), expression); } CYNumber *CYFalse::Number(CYContext &context) { @@ -732,21 +764,25 @@ CYExpression *CYNumber::PropertyName(CYContext &context) { return String(context); } -CYTarget *CYObject::Replace(CYContext &context) { +CYTarget *CYObject::Replace(CYContext &context, CYTarget *seed) { CYBuilder builder; if (properties_ != NULL) - properties_ = properties_->ReplaceAll(context, builder, $ CYThis(), false); + properties_ = properties_->ReplaceAll(context, builder, $ CYThis(), seed != this); if (builder) { return $C1($M($ CYFunctionExpression(NULL, builder.bindings_->Parameter(context), builder.statements_ ->* $ CYReturn($ CYThis()) - ), $S("call")), this, builder.bindings_->Argument(context)); + ), $S("call")), seed, builder.bindings_->Argument(context)); } CYForEach (property, properties_) property->Replace(context); - return this; + return seed; +} + +CYTarget *CYObject::Replace(CYContext &context) { + return Replace(context, this); } CYTarget *CYParenthetical::Replace(CYContext &context) { @@ -895,7 +931,7 @@ CYStatement *CYReturn::Replace(CYContext &context) { } CYTarget *CYRubyBlock::Replace(CYContext &context) { - return call_->AddArgument(context, proc_->Replace(context)); + return lhs_->AddArgument(context, proc_->Replace(context)); } CYTarget *CYRubyBlock::AddArgument(CYContext &context, CYExpression *value) { @@ -951,7 +987,10 @@ CYIdentifierFlags *CYScope::Declare(CYContext &context, CYIdentifier *identifier else if (existing->kind_ == CYIdentifierGlobal || existing->kind_ == CYIdentifierMagic) existing->kind_ = kind; else if (existing->kind_ == CYIdentifierLexical || kind == CYIdentifierLexical) - _assert(false); // XXX: throw new SyntaxError() + _assert(false); + else if (transparent_ && existing->kind_ == CYIdentifierArgument && kind == CYIdentifierVariable) + _assert(false); + // XXX: throw new SyntaxError() instead of these asserts return existing; } @@ -989,10 +1028,6 @@ void CYScope::Close(CYContext &context) { CYForEach (i, internal_) { _assert(i->identifier_->next_ == i->identifier_); switch (i->kind_) { - case CYIdentifierArgument: { - _assert(!transparent_); - } break; - case CYIdentifierLexical: { if (!damaged_) { CYIdentifier *replace(context.Unique()); @@ -1062,6 +1097,10 @@ void CYScope::Close(CYContext &context) { default:; } } } +CYTarget *CYSubscriptMember::Replace(CYContext &context) { + return $C1($M(object_, $S("$cyg")), property_); +} + CYElementValue *CYSpan::Replace(CYContext &context) { $T(NULL) return $ CYElementValue(expression_, $ CYElementValue(string_, next_->Replace(context))); } @@ -1079,6 +1118,12 @@ CYString *CYString::Concat(CYContext &context, CYString *rhs) const { return $S(value, size); } +CYIdentifier *CYString::Identifier() const { + if (const char *word = Word()) + return $ CYIdentifier(word); + return NULL; +} + CYNumber *CYString::Number(CYContext &context) { // XXX: there is a precise algorithm for this return NULL; @@ -1104,14 +1149,13 @@ CYTarget *CYStructTail::Replace(CYContext &context) { CYList names; CYForEach (field, fields_) { - CYTypedIdentifier *typed(field->typed_); - types->*$ CYElementValue(typed->Replace(context)); + types->*$ CYElementValue(field->type_->Replace(context)); CYExpression *name; - if (typed->identifier_ == NULL) + if (field->name_ == NULL) name = NULL; else - name = $S(typed->identifier_->Word()); + name = field->name_->PropertyName(context); names->*$ CYElementValue(name); } @@ -1146,6 +1190,13 @@ CYTarget *CYTemplate::Replace(CYContext &context) { return $C2($M($M($M($V("String"), $S("prototype")), $S("concat")), $S("apply")), $S(""), $ CYArray($ CYElementValue(string_, spans_->Replace(context)))); } +CYString *CYTemplate::String(CYContext &context) { + // XXX: implement this over local concat + if (spans_ != NULL) + return NULL; + return string_; +} + CYTarget *CYThis::Replace(CYContext &context) { if (context.this_ != NULL) return $V(context.this_->Identifier(context)); @@ -1211,9 +1262,19 @@ CYTarget *CYTypeConstant::Replace_(CYContext &context, CYTarget *type) { } CYStatement *CYTypeDefinition::Replace(CYContext &context) { - CYIdentifier *identifier(typed_->identifier_); - typed_->identifier_ = NULL; - return $ CYLexical(false, $B1($B(identifier, $ CYTypeExpression(typed_)))); + return $ CYLexical(false, $B1($B(name_, $ CYTypeExpression(type_)))); +} + +CYTarget *CYTypeEnum::Replace(CYContext &context) { + CYList properties; + CYForEach (constant, constants_) + properties->*$ CYPropertyValue($S(constant->name_->Word()), constant->value_); + CYObject *constants($ CYObject(properties)); + + if (specifier_ == NULL) + return $N1($V("Type"), constants); + else + return $C1($M(specifier_->Replace(context), $S("enumFor")), constants); } CYTarget *CYTypeError::Replace(CYContext &context) { @@ -1225,6 +1286,19 @@ CYTarget *CYTypeExpression::Replace(CYContext &context) { return typed_->Replace(context); } +CYTarget *CYTypeFloating::Replace(CYContext &context) { + switch (length_) { + case 0: return $V("float"); + case 1: return $V("double"); + case 2: return $V("longdouble"); + default: _assert(false); + } +} + +CYTarget *CYTypeInt128::Replace(CYContext &context) { + return $V(signing_ == CYTypeUnsigned ? "uint128" : "int128"); +} + CYTarget *CYTypeIntegral::Replace(CYContext &context) { bool u(signing_ == CYTypeUnsigned); switch (length_) { @@ -1252,7 +1326,14 @@ CYTarget *CYTypePointerTo::Replace_(CYContext &context, CYTarget *type) { } CYTarget *CYTypeReference::Replace(CYContext &context) { - return $V($pool.strcat(name_->Word(), "$cy", NULL)); + const char *prefix; + switch (kind_) { + case CYTypeReferenceStruct: prefix = "$cys"; break; + case CYTypeReferenceEnum: prefix = "$cye"; break; + default: _assert(false); + } + + return $V($pool.strcat(prefix, name_->Word(), NULL)); } CYTarget *CYTypeStruct::Replace(CYContext &context) { @@ -1274,11 +1355,11 @@ CYTarget *CYTypeVolatile::Replace_(CYContext &context, CYTarget *type) { return next_->Replace(context, $ CYCall($ CYDirectMember(type, $ CYString("volatile")))); } -CYTarget *CYTypedIdentifier::Replace(CYContext &context) { +CYTarget *CYType::Replace(CYContext &context) { return modifier_->Replace(context, specifier_->Replace(context)); } -CYTypeFunctionWith *CYTypedIdentifier::Function() { +CYTypeFunctionWith *CYType::Function() { CYTypeModifier *&modifier(CYGetLast(modifier_)); if (modifier == NULL) return NULL; @@ -1292,15 +1373,15 @@ CYTypeFunctionWith *CYTypedIdentifier::Function() { } CYArgument *CYTypedParameter::Argument(CYContext &context) { $T(NULL) - return $ CYArgument(typed_->Replace(context), next_->Argument(context)); + return $ CYArgument(type_->Replace(context), next_->Argument(context)); } CYFunctionParameter *CYTypedParameter::Parameters(CYContext &context) { $T(NULL) - return $ CYFunctionParameter($ CYBinding(typed_->identifier_ ?: context.Unique()), next_->Parameters(context)); + return $ CYFunctionParameter($ CYBinding(name_ ?: context.Unique()), next_->Parameters(context)); } CYExpression *CYTypedParameter::TypeSignature(CYContext &context, CYExpression *prefix) { $T(prefix) - return next_->TypeSignature(context, $ CYAdd(prefix, typed_->Replace(context))); + return next_->TypeSignature(context, $ CYAdd(prefix, type_->Replace(context))); } CYForInitializer *CYVar::Replace(CYContext &context) {