public override Type GenCode(ILGenerator ilGenerator, TypeBuilder parent, FieldBuilder IsBreakFields, List <FieldBuilder> level, Label endLabel)
        {
            FieldBuilder field = parent.DefineField(Name, v.Var_type.MyType(), FieldAttributes.Public);

            ilGenerator.Emit(OpCodes.Ldarg_0);                                 //meto el this
            Asig.GenCode(ilGenerator, parent, IsBreakFields, level, endLabel); //meto el valor a asignar en la pila
            ilGenerator.Emit(OpCodes.Stfld, field);                            //asigna el valor del argumento al campo field
            v.VarbBuilder = field;                                             //guardo el FieldBuilder de la variable para usarlo en el GenCode de VariableNode

            return(null);
        }
示例#2
0
        public override Type GenCode(ILGenerator ilGenerator, TypeBuilder parent, FieldBuilder IsBreakFields, List <FieldBuilder> level, Label endLabel)
        {
            //creo una clase para el for
            TypeBuilder        forClass       = parent.DefineNestedType("For" + Utils.AutoNumeric, TypeAttributes.NestedPublic | TypeAttributes.Class); //defino la clase anidada
            ConstructorBuilder forConstructor = forClass.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, new List <Type>()
            {
                parent, typeof(int), typeof(int)
            }.ToArray());
            ILGenerator ilFor = forConstructor.GetILGenerator(); //cojo el generador de la clase del for

            //creando el campo 'parent'
            FieldBuilder forParent = forClass.DefineField("parent", parent, FieldAttributes.Public); //defino el campo parent del for

            ilFor.Emit(OpCodes.Ldarg_0);                                                             //meto el this
            ilFor.Emit(OpCodes.Ldarg_1);                                                             //meto el padre en la pila
            ilFor.Emit(OpCodes.Stfld, forParent);                                                    //el padre ahora es el for (this.parent = forParent)

            //creando el campo 'Break'
            FieldBuilder field = forClass.DefineField("Break", typeof(bool), FieldAttributes.Public);

            ilFor.Emit(OpCodes.Ldarg_0);      //meto el this
            ilFor.Emit(OpCodes.Ldc_I4_0);     //meto un cero en la pila
            ilFor.Emit(OpCodes.Stfld, field); //crea el campo con false

            //creando el campo del i
            FieldBuilder index = forClass.DefineField(Name, typeof(int), FieldAttributes.Public); //i del for (de la clase)

            ilFor.Emit(OpCodes.Ldarg_0);                                                          //meto el this
            ilFor.Emit(OpCodes.Ldarg_2);                                                          //meto el padre en la pila
            ilFor.Emit(OpCodes.Stfld, index);                                                     //el padre ahora es el for (this.parent = forParent)
            //Type asignType = Asig.GenCode(ilFor, forClass, field,level, endLabel); //genero el codigo de la asignacion (mete la asignacion en la pila)

            //ilFor.Emit(OpCodes.Stfld, index); //actualizo el valor de la asignacion
            varFor.VarbBuilder = index;

            //creando el to del for
            FieldBuilder toBuilder = forClass.DefineField("To" + Utils.AutoNumeric, typeof(int), FieldAttributes.Public);

            //LocalBuilder to = ilFor.DeclareLocal(typeof(int)); //declaro el to del for (para quedarme con el valor)

            ilFor.Emit(OpCodes.Ldarg_0);          //meto el this
            ilFor.Emit(OpCodes.Ldarg_3);          //meto el padre en la pila
            ilFor.Emit(OpCodes.Stfld, toBuilder); //el padre ahora es el for (this.parent = forParent)
            //Type toType = To.GenCode(ilFor, forClass, field, level, endLabel); //en el tope esta el valor del to

            //ilFor.Emit(OpCodes.Stfld, toBuilder); //actualizo el valor de la asignacion

            //Cerrar el constructor
            ilFor.Emit(OpCodes.Ret);

            //creo el metodo donde se encuentra el body
            MethodBuilder bodyBuilder = forClass.DefineMethod("ForBody", MethodAttributes.Public, CallingConventions.HasThis, typeof(void), new Type[] { });
            ILGenerator   ilBody      = bodyBuilder.GetILGenerator();

            //comienza la generacion de codigo del for
            Label start = ilBody.DefineLabel();
            Label end   = ilBody.DefineLabel();

            //comienza el for
            //meto en el tope el campo this.index
            ilBody.MarkLabel(start);           //defino el comienzo del for
            ilBody.Emit(OpCodes.Ldarg_0);      //meto el this
            ilBody.Emit(OpCodes.Ldfld, index); //meto la i del for

            //meto en el campo this.toBuilder
            ilBody.Emit(OpCodes.Ldarg_0);          //meto el this
            ilBody.Emit(OpCodes.Ldfld, toBuilder); //meto el to del for
            ilBody.Emit(OpCodes.Bgt, end);         //si i > to se acabo el for
            level.Add(forParent);

            //genero el codigo del body
            Type bodyType = Body.GenCode(ilBody, forClass, field, level, end);

            level.RemoveAt(level.Count - 1);

            ilBody.Emit(OpCodes.Ldarg_0);
            ilBody.Emit(OpCodes.Ldfld, field);
            ilBody.Emit(OpCodes.Ldc_I4_1);
            ilBody.Emit(OpCodes.Beq, end);

            //actualizo this.index
            ilBody.Emit(OpCodes.Ldarg_0);      //meto el this
            ilBody.Emit(OpCodes.Ldarg_0);      //meto el this
            ilBody.Emit(OpCodes.Ldfld, index); //busco el valor de this.i y lo dejo en el tope
            ilBody.Emit(OpCodes.Ldc_I4_1);     //meto el 1

            ilBody.Emit(OpCodes.Add);          //i + 1
            ilBody.Emit(OpCodes.Stfld, index); //i = i + 1 (uso el replace estatico porque lo que tengo en el tope es un valor y no una referencia)
            ilBody.Emit(OpCodes.Br, start);    //comienza de nuevo el for

            ilBody.MarkLabel(end);             //final

            ilBody.Emit(OpCodes.Ret);          //se acabo el metodo del body del for

            forClass.CreateType();             //creo el tipo de la clase for

            // Crear una instancia del tipo que representa al for.
            ilGenerator.Emit(OpCodes.Ldarg_0);
            Asig.GenCode(ilGenerator, parent, IsBreakFields, level, endLabel);
            To.GenCode(ilGenerator, parent, IsBreakFields, level, endLabel);
            ilGenerator.Emit(OpCodes.Newobj, forConstructor);

            // Llamar al método que representa el cuerpo del for.
            ilGenerator.Emit(OpCodes.Callvirt, bodyBuilder);
            return(null); //el for no retorna valor
        }