Ejemplo n.º 1
0
        protected override void EmitRead(Compiler.CompilerContext ctx, Compiler.Local valueFrom)
        {
            using (Compiler.Local oldList = AppendToCollection ? ctx.GetLocalWithValue(ExpectedType, valueFrom) : null)
                using (Compiler.Local builder = new Compiler.Local(ctx, builderFactory.ReturnType))
                {
                    ctx.EmitCall(builderFactory);
                    ctx.StoreValue(builder);

                    if (AppendToCollection)
                    {
                        Compiler.CodeLabel done = ctx.DefineLabel();
                        if (!Helpers.IsValueType(ExpectedType))
                        {
                            ctx.LoadValue(oldList);
                            ctx.BranchIfFalse(done, false); // old value null; nothing to add
                        }
#if COREFX
                        TypeInfo typeInfo = ExpectedType.GetTypeInfo();
#else
                        Type typeInfo = ExpectedType;
#endif
                        PropertyInfo prop = Helpers.GetProperty(typeInfo, "Length", false);
                        if (prop == null)
                        {
                            prop = Helpers.GetProperty(typeInfo, "Count", false);
                        }
#if !NO_GENERICS
                        if (prop == null)
                        {
                            prop = Helpers.GetProperty(ResolveIReadOnlyCollection(ExpectedType, Tail.ExpectedType), "Count", false);
                        }
#endif
                        ctx.LoadAddress(oldList, oldList.Type);
                        ctx.EmitCall(Helpers.GetGetMethod(prop, false, false));
                        ctx.BranchIfFalse(done, false); // old list is empty; nothing to add

                        Type voidType = ctx.MapType(typeof(void));
                        if (addRange != null)
                        {
                            ctx.LoadValue(builder);
                            ctx.LoadValue(oldList);
                            ctx.EmitCall(addRange);
                            if (addRange.ReturnType != null && add.ReturnType != voidType)
                            {
                                ctx.DiscardValue();
                            }
                        }
                        else
                        {
                            // loop and call Add repeatedly
                            MethodInfo moveNext, current, getEnumerator = GetEnumeratorInfo(ctx.Model, out moveNext, out current);
                            Helpers.DebugAssert(moveNext != null);
                            Helpers.DebugAssert(current != null);
                            Helpers.DebugAssert(getEnumerator != null);

                            Type enumeratorType = getEnumerator.ReturnType;
                            using (Compiler.Local iter = new Compiler.Local(ctx, enumeratorType))
                            {
                                ctx.LoadAddress(oldList, ExpectedType);
                                ctx.EmitCall(getEnumerator);
                                ctx.StoreValue(iter);
                                using (ctx.Using(iter))
                                {
                                    Compiler.CodeLabel body = ctx.DefineLabel(), next = ctx.DefineLabel();
                                    ctx.Branch(next, false);

                                    ctx.MarkLabel(body);
                                    ctx.LoadAddress(builder, builder.Type);
                                    ctx.LoadAddress(iter, enumeratorType);
                                    ctx.EmitCall(current);
                                    ctx.EmitCall(add);
                                    if (add.ReturnType != null && add.ReturnType != voidType)
                                    {
                                        ctx.DiscardValue();
                                    }

                                    ctx.MarkLabel(@next);
                                    ctx.LoadAddress(iter, enumeratorType);
                                    ctx.EmitCall(moveNext);
                                    ctx.BranchIfTrue(body, false);
                                }
                            }
                        }


                        ctx.MarkLabel(done);
                    }

                    EmitReadList(ctx, builder, Tail, add, packedWireType, false);

                    ctx.LoadAddress(builder, builder.Type);
                    ctx.EmitCall(finish);
                    if (ExpectedType != finish.ReturnType)
                    {
                        ctx.Cast(ExpectedType);
                    }
                }
        }
Ejemplo n.º 2
0
        protected override void EmitRead(Compiler.CompilerContext ctx, Compiler.Local valueFrom)
        {
            using (Compiler.Local oldList = AppendToCollection ? ctx.GetLocalWithValue(ExpectedType, valueFrom) : null)
                using (Compiler.Local builder = new Compiler.Local(ctx, builderFactory.ReturnType))
                {
                    ctx.EmitCall(builderFactory);
                    ctx.StoreValue(builder);

                    if (AppendToCollection)
                    {
                        Compiler.CodeLabel done = ctx.DefineLabel();
                        if (!Helpers.IsValueType(ExpectedType))
                        {
                            ctx.LoadValue(oldList);
                            ctx.BranchIfFalse(done, false); // old value null; nothing to add
                        }

                        ctx.LoadAddress(oldList, oldList.Type);
                        if (isEmpty != null)
                        {
                            ctx.EmitCall(Helpers.GetGetMethod(isEmpty, false, false));
                            ctx.BranchIfTrue(done, false); // old list is empty; nothing to add
                        }
                        else
                        {
                            ctx.EmitCall(Helpers.GetGetMethod(length, false, false));
                            ctx.BranchIfFalse(done, false); // old list is empty; nothing to add
                        }

                        Type voidType = typeof(void);
                        if (addRange != null)
                        {
                            ctx.LoadValue(builder);
                            ctx.LoadValue(oldList);
                            ctx.EmitCall(addRange);
                            if (addRange.ReturnType != null && add.ReturnType != voidType)
                            {
                                ctx.DiscardValue();
                            }
                        }
                        else
                        {
                            // loop and call Add repeatedly
                            MethodInfo moveNext, current, getEnumerator = GetEnumeratorInfo(out moveNext, out current);
                            Helpers.DebugAssert(moveNext != null);
                            Helpers.DebugAssert(current != null);
                            Helpers.DebugAssert(getEnumerator != null);

                            Type enumeratorType = getEnumerator.ReturnType;
                            using (Compiler.Local iter = new Compiler.Local(ctx, enumeratorType))
                            {
                                ctx.LoadAddress(oldList, ExpectedType);
                                ctx.EmitCall(getEnumerator);
                                ctx.StoreValue(iter);
                                using (ctx.Using(iter))
                                {
                                    Compiler.CodeLabel body = ctx.DefineLabel(), next = ctx.DefineLabel();
                                    ctx.Branch(next, false);

                                    ctx.MarkLabel(body);
                                    ctx.LoadAddress(builder, builder.Type);
                                    ctx.LoadAddress(iter, enumeratorType);
                                    ctx.EmitCall(current);
                                    ctx.EmitCall(add);
                                    if (add.ReturnType != null && add.ReturnType != voidType)
                                    {
                                        ctx.DiscardValue();
                                    }

                                    ctx.MarkLabel(@next);
                                    ctx.LoadAddress(iter, enumeratorType);
                                    ctx.EmitCall(moveNext);
                                    ctx.BranchIfTrue(body, false);
                                }
                            }
                        }

                        ctx.MarkLabel(done);
                    }

                    EmitReadList(ctx, builder, Tail, add, packedWireType, false);

                    ctx.LoadAddress(builder, builder.Type);
                    ctx.EmitCall(finish);
                    if (ExpectedType != finish.ReturnType)
                    {
                        ctx.Cast(ExpectedType);
                    }
                }
        }
Ejemplo n.º 3
0
        protected override void EmitRead(Compiler.CompilerContext ctx, Compiler.Local valueFrom)
        {
            using (ctx.StartDebugBlockAuto(this))
            {
                Type voidType = ctx.MapType(typeof(void));
                using (Compiler.Local value = ctx.GetLocalWithValueForEmitRead(this, valueFrom))
                    using (Compiler.Local builderInstance = new Compiler.Local(ctx, _builderFactory.ReturnType))
                        using (Compiler.Local trappedKey = new Compiler.Local(ctx, typeof(int)))
                        {
                            ctx.G.Assign(trappedKey, ctx.G.ReaderFunc.ReserveNoteObject_int());
                            ctx.EmitCall(_builderFactory);
                            ctx.StoreValue(builderInstance);

                            if (AppendToCollection)
                            {
                                Compiler.CodeLabel done = ctx.DefineLabel();
                                if (!ExpectedType.IsValueType)
                                {
                                    ctx.LoadValue(value);
                                    ctx.BranchIfFalse(done, false); // old value null; nothing to add
                                }
                                PropertyInfo prop = Helpers.GetProperty(ExpectedType, "Length", false) ?? Helpers.GetProperty(ExpectedType, "Count", false);
#if !NO_GENERICS
                                if (prop == null)
                                {
                                    prop = Helpers.GetProperty(ResolveIReadOnlyCollection(ExpectedType, Tail.ExpectedType), "Count", false);
                                }
#endif
                                ctx.LoadAddress(value, value.Type);
                                ctx.EmitCall(Helpers.GetGetMethod(prop, false, false));
                                ctx.BranchIfFalse(done, false); // old list is empty; nothing to add

                                if (_addRange != null)
                                {
                                    ctx.LoadValue(builderInstance);
                                    ctx.LoadValue(value);
                                    ctx.EmitCall(_addRange);
                                    if (_addRange.ReturnType != null && _add.ReturnType != voidType)
                                    {
                                        ctx.DiscardValue();
                                    }
                                }
                                else
                                {
                                    // loop and call Add repeatedly
                                    MethodInfo moveNext, current, getEnumerator = GetEnumeratorInfo(ctx.Model, out moveNext, out current);
                                    Helpers.DebugAssert(moveNext != null);
                                    Helpers.DebugAssert(current != null);
                                    Helpers.DebugAssert(getEnumerator != null);

                                    Type enumeratorType = getEnumerator.ReturnType;
                                    using (Compiler.Local iter = new Compiler.Local(ctx, enumeratorType))
                                    {
                                        ctx.LoadAddress(value, ExpectedType);
                                        ctx.EmitCall(getEnumerator);
                                        ctx.StoreValue(iter);
                                        using (ctx.Using(iter))
                                        {
                                            Compiler.CodeLabel body = ctx.DefineLabel(), next = ctx.DefineLabel();
                                            ctx.Branch(next, false);

                                            ctx.MarkLabel(body);
                                            ctx.LoadAddress(builderInstance, builderInstance.Type);
                                            ctx.LoadAddress(iter, enumeratorType);
                                            ctx.EmitCall(current);
                                            ctx.EmitCall(_add);
                                            if (_add.ReturnType != null && _add.ReturnType != voidType)
                                            {
                                                ctx.DiscardValue();
                                            }

                                            ctx.MarkLabel(@next);
                                            ctx.LoadAddress(iter, enumeratorType);
                                            ctx.EmitCall(moveNext);
                                            ctx.BranchIfTrue(body, false);
                                        }
                                    }
                                }


                                ctx.MarkLabel(done);
                            }

                            ListHelpers.EmitRead(
                                ctx.G,
                                null,
                                null,
                                o =>
                            {
                                using (ctx.StartDebugBlockAuto(this, "add"))
                                {
                                    ctx.LoadAddress(builderInstance, builderInstance.Type);
                                    ctx.LoadValue(o);
                                    ctx.EmitCall(_add);
                                    if (_add.ReturnType != null && _add.ReturnType != voidType)
                                    {
                                        ctx.DiscardValue();
                                    }
                                }
                            });

                            ctx.LoadAddress(builderInstance, builderInstance.Type);
                            ctx.EmitCall(_finish);
                            if (ExpectedType != _finish.ReturnType)
                            {
                                ctx.Cast(ExpectedType);
                            }
                            ctx.StoreValue(value);
                            ctx.G.Reader.NoteReservedTrappedObject(trappedKey, value);

                            if (EmitReadReturnsValue)
                            {
                                ctx.LoadValue(value);
                            }
                        }
            }
        }