Index: src/Validate/Autogen.cpp
===================================================================
--- src/Validate/Autogen.cpp	(revision a55ebcca1b712f00e2fd94a9d348511dd62c8f8a)
+++ src/Validate/Autogen.cpp	(revision 0522ebe7b3ae7204a1d2c50a1bc7273bfa36762a)
@@ -206,4 +206,5 @@
 	ast::FunctionDecl * genLabelProto() const;
 	ast::FunctionDecl * genValueProto() const;
+	// ast::FunctionDecl * genValueProto2() const;
 };
 
@@ -790,9 +791,17 @@
 }
 
+// ast::FunctionDecl * EnumFuncGenerator::genValueProto2() const {
+// 	return genProto( "valueE", 
+// 		{ new ast::ObjectDecl( getLocation(), "_i", new ast::EnumPosType( new ast::EnumInstType( decl ) ) )},
+// 		{ new ast::ObjectDecl( getLocation(), "_ret", ast::deepCopy( decl->base ) ) } );
+// }
+
 void EnumFuncGenerator::genAttrFuncForward() {	
 	if ( decl->base ) {
 		ast::FunctionDecl *(EnumFuncGenerator::*attrProtos[3])() const = {
 			&EnumFuncGenerator::genPosProto, &EnumFuncGenerator::genLabelProto, 
-			&EnumFuncGenerator::genValueProto };
+			&EnumFuncGenerator::genValueProto
+			// , &EnumFuncGenerator::genValueProto2 
+			};
 		for ( auto & generator : attrProtos ) {
 			produceForwardDecl( (this->*generator)() );
Index: src/Validate/ReplacePseudoFunc.cpp
===================================================================
--- src/Validate/ReplacePseudoFunc.cpp	(revision a55ebcca1b712f00e2fd94a9d348511dd62c8f8a)
+++ src/Validate/ReplacePseudoFunc.cpp	(revision 0522ebe7b3ae7204a1d2c50a1bc7273bfa36762a)
@@ -6,13 +6,11 @@
 #include "AST/Inspect.hpp"
 #include "AST/Pass.hpp"
+#include "AST/Print.hpp"
 #include "AST/Stmt.hpp"
 #include "Common/utility.h"
+#include "ResolvExpr/CandidateFinder.hpp"
 #include "ResolvExpr/Resolver.h"
 #include "SymTab/Mangler.h"
 
-#include "AST/Print.hpp"
-
-#include "ResolvExpr/CandidateFinder.hpp"
-
 namespace Validate {
 
@@ -21,4 +19,20 @@
 std::set<std::string> queryLabels;
 std::set<std::string> queryValues;
+
+struct ReplaceEnumInstWithPos final : public ast::WithShortCircuiting {
+    void previsit(const ast::ObjectDecl*) { visit_children = false; }
+    const ast::ObjectDecl* postvisit(const ast::ObjectDecl* decl) {
+        auto enumInst = decl->type.strict_as<ast::EnumInstType>();
+        auto enumPos = new ast::EnumPosType(enumInst);
+        auto ret = ast::mutate_field(decl, &ast::ObjectDecl::type, enumPos);
+        ret = ast::mutate_field(ret, &ast::ObjectDecl::mangleName,
+                                Mangle::mangle(ret));
+        return ret;
+    }
+};
+
+const inline std::string getValueArrayName(std::string enumName) {
+    return "values_" + enumName;
+}
 
 // struct AutoInit {
@@ -32,5 +46,5 @@
     void previsit(const ast::ApplicationExpr* expr);
     void previsit(const ast::CastExpr* expr);
-    void previsit(const ast::VariableExpr* ) { visit_children = false; }
+    void previsit(const ast::VariableExpr*) { visit_children = false; }
 
     ast::Expr const* postvisit(const ast::VariableExpr* expr);
@@ -66,8 +80,7 @@
     auto varExpr = expr->func.as<ast::VariableExpr>();
     auto fname = ast::getFunctionName(expr);
-	if ( !varExpr || varExpr->var->linkage == ast::Linkage::Intrinsic ) {
-        if ( fname == "?{}" || fname == "?=?" )
-		    visit_children = false;
-	}
+    if (!varExpr || varExpr->var->linkage == ast::Linkage::Intrinsic) {
+        if (fname == "?{}" || fname == "?=?") visit_children = false;
+    }
 
     if (fname == "labelE" || fname == "valueE" || fname == "posE") {
@@ -94,10 +107,10 @@
             auto untyped = new ast::UntypedExpr(
                 expr->location, new ast::NameExpr(expr->location, "valueE"),
-                { std::move( expr ) });
+                {std::move(expr)});
             ResolvExpr::ResolveContext context{symtab, transUnit().global};
             auto result = ResolvExpr::findVoidExpression(untyped, context);
             ast::ptr<ast::ApplicationExpr> ret =
-                    result.strict_as<ast::ApplicationExpr>();
-            return ast::deepCopy( ret );
+                result.strict_as<ast::ApplicationExpr>();
+            return ast::deepCopy(ret);
         }
     }
@@ -132,27 +145,24 @@
 }
 
-const ast::Init * getAutoInit( const CodeLocation & location,
-    const ast::Type * type, ResolvExpr::ResolveContext context, const ast::Init * prev ) {
-    if ( auto prevInit = dynamic_cast< const ast::SingleInit * >( prev ) ) {
+const ast::Init* getAutoInit(const CodeLocation& location,
+                             const ast::Type* type,
+                             ResolvExpr::ResolveContext context,
+                             const ast::Init* prev) {
+    if (auto prevInit = dynamic_cast<const ast::SingleInit*>(prev)) {
         auto prevInitExpr = prevInit->value;
-        if ( auto constInit = prevInitExpr.as< ast::ConstantExpr >() ) {
+        if (auto constInit = prevInitExpr.as<ast::ConstantExpr>()) {
             // Assume no string literal for now
             return new ast::SingleInit(
-                location,
-                ast::ConstantExpr::from_int(
-                    location, constInit->intValue() + 1 )
-            );
+                location, ast::ConstantExpr::from_int(
+                              location, constInit->intValue() + 1));
         } else {
             auto untypedThisInit = new ast::UntypedExpr(
-                    location,
-                    new ast::NameExpr( location, "?++" ),
-                    { prevInitExpr }
-                );
-            auto typedInit = ResolvExpr::findSingleExpression(untypedThisInit, type,
-                context );
-            return new ast::SingleInit( location, typedInit );
-        }
-    }
-    SemanticError( prev, "Auto Init a List is not implemented" );
+                location, new ast::NameExpr(location, "?++"), {prevInitExpr});
+            auto typedInit = ResolvExpr::findSingleExpression(untypedThisInit,
+                                                              type, context);
+            return new ast::SingleInit(location, typedInit);
+        }
+    }
+    SemanticError(prev, "Auto Init a List is not implemented");
     return prev;
 }
@@ -167,26 +177,25 @@
     auto type = enumDecl->base;
 
-    for ( size_t i = 0; i < enumDecl->members.size(); i++ ) {
-        ast::ptr<ast::Decl> mem = enumDecl->members.at( i );
-        auto memAsObjectDecl = mem.as< ast::ObjectDecl >();
-        assert( memAsObjectDecl );
-        if ( memAsObjectDecl->init ) {
-            inits.emplace_back( memAsObjectDecl->init );
+    for (size_t i = 0; i < enumDecl->members.size(); i++) {
+        ast::ptr<ast::Decl> mem = enumDecl->members.at(i);
+        auto memAsObjectDecl = mem.as<ast::ObjectDecl>();
+        assert(memAsObjectDecl);
+        if (memAsObjectDecl->init) {
+            inits.emplace_back(memAsObjectDecl->init);
         } else {
-            const CodeLocation & location = mem->location;
-            if ( i == 0 ) {
-                inits.emplace_back( new ast::SingleInit( 
-                    location, 
-                    ast::ConstantExpr::from_int( mem->location, 0 )
-                ) );
+            const CodeLocation& location = mem->location;
+            if (i == 0) {
+                inits.emplace_back(new ast::SingleInit(
+                    location, ast::ConstantExpr::from_int(mem->location, 0)));
             } else {
-                inits.emplace_back( getAutoInit( location, enumDecl->base, 
-                    ResolvExpr::ResolveContext{symtab, transUnit().global}, 
-                    inits.at( i - 1 ).as<ast::SingleInit>()) );
+                inits.emplace_back(getAutoInit(
+                    location, enumDecl->base,
+                    ResolvExpr::ResolveContext{symtab, transUnit().global},
+                    inits.at(i - 1).as<ast::SingleInit>()));
             }
         }
         labels.emplace_back(new ast::SingleInit(
-        location, ast::ConstantExpr::from_string(location, mem->name)));
-    } 
+            location, ast::ConstantExpr::from_string(location, mem->name)));
+    }
     if (queryValues.count(enumDecl->name)) {
         auto init = new ast::ListInit(location, std::move(inits));
@@ -219,17 +228,34 @@
 ast::ApplicationExpr const* getPseudoFuncApplication(
     const CodeLocation location, ResolvExpr::ResolveContext context,
-    const ast::VariableExpr* arg, const ast::EnumDecl* base, const std::string & name) {
+    const ast::VariableExpr* arg, const ast::EnumDecl* base,
+    const std::string& name) {
     ast::Expr* toResolve = new ast::NameExpr(location, name + base->name);
-    auto result = ResolvExpr::findVoidExpression(toResolve, context);
-    assert(result.get());
-    auto arrAsVar = result.strict_as<ast::VariableExpr>();
-    auto untyped = new ast::UntypedExpr(
-        location, new ast::NameExpr(location, "?[?]"),
-        { std::move(arrAsVar), std::move(arg) });
-    auto typedResult = ResolvExpr::findVoidExpression(untyped, context);
-
-    ast::ptr<ast::ApplicationExpr> ret =
-        typedResult.strict_as<ast::ApplicationExpr>();
-    return ast::deepCopy(ret);
+    // Find the request arrary
+    auto arr = ResolvExpr::findVoidExpression(toResolve, context);
+    assert(arr.get());
+    auto arrAsVar = arr.strict_as<ast::VariableExpr>();
+    // change EnumInstType to EnumPosType to avoid recursive resolution
+    auto argAsDecl = arg->var.as<ast::ObjectDecl>();
+    if (argAsDecl->type.as<ast::EnumInstType>()) {
+        ast::Pass<ReplaceEnumInstWithPos> replacer;
+        auto rep = argAsDecl->accept(replacer);
+        auto mutatedArg = ast::mutate_field(arg, &ast::VariableExpr::var, rep);
+        auto untyped =
+            new ast::UntypedExpr(location, new ast::NameExpr(location, "?[?]"),
+                                 {std::move(arrAsVar), mutatedArg});
+        auto typedResult = ResolvExpr::findVoidExpression(untyped, context);
+        ast::ptr<ast::ApplicationExpr> ret =
+            typedResult.strict_as<ast::ApplicationExpr>();
+        return ast::deepCopy(ret);
+    } else {
+        auto untyped =
+            new ast::UntypedExpr(location, new ast::NameExpr(location, "?[?]"),
+                                 {std::move(arrAsVar), arg});
+        auto typedResult = ResolvExpr::findVoidExpression(untyped, context);
+
+        ast::ptr<ast::ApplicationExpr> ret =
+            typedResult.strict_as<ast::ApplicationExpr>();
+        return ast::deepCopy(ret);
+    }
 }
 
@@ -250,45 +276,107 @@
         const ast::ObjectDecl* argAsVar = arg->var.as<const ast::ObjectDecl>();
         const std::string referredName = argAsVar->name;
-        const ast::EnumInstType* argType =
-            argAsVar->type.as<const ast::EnumInstType>();
-        if (!argType) {
+
+        if (const ast::EnumInstType* argTypeAsEnumInst =
+                argAsVar->type.as<const ast::EnumInstType>()) {
+            const ast::EnumDecl* base = argTypeAsEnumInst->base;
+            ResolvExpr::ResolveContext context{symtab, transUnit().global};
+            // If resolvable as constant
+            for (size_t i = 0; i < base->members.size(); i++) {
+                if (base->members[i]->name == referredName) {
+                    if (fname == "posE")
+                        return ast::ConstantExpr::from_int(expr->location, i);
+                    else if (fname == "labelE")
+                        return ast::ConstantExpr::from_string(expr->location,
+                                                              referredName);
+                    else {
+                        return getPseudoFuncApplication(
+                            location, context, arg.get(), base, "values_");
+                    }
+                }
+            }
+
+            if (fname == "labelE") {
+                if (auto labelExpr = getPseudoFuncApplication(
+                        location, context, arg.get(), base, "labels_")) {
+                    return labelExpr;
+                }
+            } else if (fname == "valueE") {
+                if (auto valueExpr = getPseudoFuncApplication(
+                        location, context, arg.get(), base, "values_")) {
+                    return valueExpr;
+                }
+            } else {  // it is position; replace itself
+                return std::move(arg.get());
+            }
+        } else if (const ast::EnumPosType* argTypeAsPos =
+                       argAsVar->type.as<const ast::EnumPosType>()) {
+            const ast::EnumDecl* base = argTypeAsPos->instance->base;
+            ResolvExpr::ResolveContext context{symtab, transUnit().global};
+            if (fname == "labelE") {
+                if (auto labelExpr = getPseudoFuncApplication(
+                        location, context, arg.get(), base, "labels_")) {
+                    return labelExpr;
+                }
+            } else if (fname == "valueE") {
+                if (auto valueExpr = getPseudoFuncApplication(
+                        location, context, arg.get(), base, "values_")) {
+                    return valueExpr;
+                }
+            } else {  // it is position; replace itself
+                return std::move(arg.get());
+            }
+        } else {
             SemanticError(argAsVar,
                           "Pseudo Enum Expression can only be used on an "
                           "enumeration instance");
         }
-        const ast::EnumDecl* base = argType->base;
-        ResolvExpr::ResolveContext context{symtab, transUnit().global};
-        // If resolvable as constant
-        for (size_t i = 0; i < base->members.size(); i++) {
-            if (base->members[i]->name == referredName) {
-                if (fname == "posE")
-                    return ast::ConstantExpr::from_int(expr->location, i);
-                else if (fname == "labelE")
-                    return ast::ConstantExpr::from_string(expr->location,
-                                                          referredName);
-                else {
-                    return getPseudoFuncApplication(location, context, arg.get(),
-                                               base, "values_");                    
-                }
-
-            }
-        }
-
-        if (fname == "labelE") {
-            if (auto labelExpr =
-                    getPseudoFuncApplication(location, context, arg.get(), base, "labels_")) {
-                return labelExpr;
-            }
-        } else if (fname == "valueE") {
-            if (auto valueExpr =
-                    getPseudoFuncApplication(location, context, arg.get(), base, "values_")) {
-                return valueExpr;
-            }
-        } else { // it is position; replace itself
-            return std::move( arg.get() );
-        }
     }
     return expr;
 }
+
+ast::ptr<ast::Expr> reduceCastExpr(ast::ptr<ast::Expr> expr) {
+    if (auto castExpr = expr.as<ast::CastExpr>()) {
+        return reduceCastExpr(castExpr->arg);
+    }
+    return expr;
+}
+
+struct ReplaceEnumInst final {
+    const ast::Expr* postvisit(const ast::ApplicationExpr* expr) {
+        auto fname = ast::getFunctionName(expr);
+        if (fname != "?[?]") return expr;
+        if (expr->args.size() != 2) return expr;
+
+        auto arg1AsVar =
+            reduceCastExpr(expr->args.front()).as<ast::VariableExpr>();
+        auto arg2AsVar =
+            reduceCastExpr(expr->args.back()).as<ast::VariableExpr>();
+
+        if (!arg1AsVar || !arg2AsVar) return expr;
+
+        auto arg1Asecl = arg1AsVar->var.as<ast::ObjectDecl>();
+        auto arg2Asecl = arg2AsVar->var.as<ast::ObjectDecl>();
+
+        if (!arg1Asecl || !arg2Asecl) return expr;
+        auto arrInst = arg1Asecl->type.as<ast::ArrayType>();
+        auto pointerInst = arg1Asecl->type.as<ast::PointerType>();
+        if (!arrInst && !pointerInst) {
+            return expr;
+        }
+        auto enumInst = arg2Asecl->type.as<ast::EnumInstType>();
+        if (!enumInst) return expr;
+
+        const std::string arrName = arg1Asecl->name;
+        if (arrName != getValueArrayName(enumInst->base->name)) return expr;
+        ast::Pass<ReplaceEnumInstWithPos> replacer;
+        auto rep = arg2Asecl->accept(replacer);
+        if (!rep) return expr;
+        auto mutObj =
+            ast::mutate_field(arg2AsVar, &ast::VariableExpr::var, rep);
+        auto mut = ast::mutate_field_index(expr, &ast::ApplicationExpr::args, 1,
+                                           mutObj);
+        return mut;
+    }
+};
 
 }  // namespace
@@ -299,4 +387,5 @@
     ast::Pass<PseudoFuncGenerateRoutine>::run(translationUnit);
     ast::Pass<ReplacePseudoFuncCore>::run(translationUnit);
+    ast::Pass<ReplaceEnumInst>::run(translationUnit);
 }
 }  // namespace Validate
