diff --git a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/common/JavaUpCallsRFFIImpl.java b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/common/JavaUpCallsRFFIImpl.java
index c2b3d6e9517a0b66d9dcd7016c6648ad372f41c2..7a2924577a1ebe8a2f65a0c61e6ee89d054c99b9 100644
--- a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/common/JavaUpCallsRFFIImpl.java
+++ b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/common/JavaUpCallsRFFIImpl.java
@@ -287,21 +287,8 @@ public abstract class JavaUpCallsRFFIImpl implements UpCallsRFFI {
     }
 
     @Override
-    @TruffleBoundary
     public Object Rf_getAttrib(Object obj, Object name) {
-        Object result = RNull.instance;
-        if (obj instanceof RAttributable) {
-            RAttributable attrObj = (RAttributable) obj;
-            DynamicObject attrs = attrObj.getAttributes();
-            if (attrs != null) {
-                String nameAsString = Utils.intern(((RSymbol) name).getName());
-                Object attr = attrs.get(nameAsString);
-                if (attr != null) {
-                    result = attr;
-                }
-            }
-        }
-        return result;
+        throw implementedAsNode();
     }
 
     @Override
@@ -891,20 +878,8 @@ public abstract class JavaUpCallsRFFIImpl implements UpCallsRFFI {
     }
 
     @Override
-    @TruffleBoundary
     public Object R_tryEval(Object expr, Object env, int silent) {
-        Object handlerStack = RErrorHandling.getHandlerStack();
-        Object restartStack = RErrorHandling.getRestartStack();
-        try {
-            // TODO handle silent
-            RErrorHandling.resetStacks();
-            Object result = Rf_eval(expr, env);
-            return result;
-        } catch (Throwable t) {
-            return null;
-        } finally {
-            RErrorHandling.restoreStacks(handlerStack, restartStack);
-        }
+        throw implementedAsNode();
     }
 
     /**
diff --git a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/AttributesAccessNodes.java b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/AttributesAccessNodes.java
index 7b6ef2821fb05a7cbff327b6d1bf052e005a460c..19f9d9ea72e09bdde8aa66cb4477dce61d30f2ca 100644
--- a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/AttributesAccessNodes.java
+++ b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/AttributesAccessNodes.java
@@ -22,17 +22,24 @@
  */
 package com.oracle.truffle.r.ffi.impl.nodes;
 
+import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.singleElement;
+import static com.oracle.truffle.r.nodes.builtin.casts.fluent.CastNodeBuilder.newCastBuilder;
+
 import com.oracle.truffle.api.CompilerDirectives;
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
 import com.oracle.truffle.api.dsl.Cached;
 import com.oracle.truffle.api.dsl.Fallback;
 import com.oracle.truffle.api.dsl.Specialization;
+import com.oracle.truffle.api.profiles.ConditionProfile;
 import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodesFactory.ATTRIBNodeGen;
 import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodesFactory.CopyMostAttribNodeGen;
 import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodesFactory.TAGNodeGen;
 import com.oracle.truffle.r.nodes.attributes.CopyOfRegAttributesNode;
+import com.oracle.truffle.r.nodes.attributes.GetAttributeNode;
 import com.oracle.truffle.r.nodes.attributes.GetAttributesNode;
 import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetNamesAttributeNode;
+import com.oracle.truffle.r.nodes.function.opt.UpdateShareableChildValueNode;
+import com.oracle.truffle.r.nodes.unary.CastNode;
 import com.oracle.truffle.r.runtime.ArgumentsSignature;
 import com.oracle.truffle.r.runtime.RError;
 import com.oracle.truffle.r.runtime.RError.Message;
@@ -41,14 +48,43 @@ import com.oracle.truffle.r.runtime.RRuntime;
 import com.oracle.truffle.r.runtime.data.RArgsValuesAndNames;
 import com.oracle.truffle.r.runtime.data.RAttributable;
 import com.oracle.truffle.r.runtime.data.RAttributeStorage;
+import com.oracle.truffle.r.runtime.data.RDataFactory;
 import com.oracle.truffle.r.runtime.data.RExternalPtr;
 import com.oracle.truffle.r.runtime.data.RList;
 import com.oracle.truffle.r.runtime.data.RNull;
 import com.oracle.truffle.r.runtime.data.RPairList;
 import com.oracle.truffle.r.runtime.data.RStringVector;
+import com.oracle.truffle.r.runtime.data.RSymbol;
 
 public final class AttributesAccessNodes {
 
+    public static final class GetAttrib extends FFIUpCallNode.Arg2 {
+        @Child private GetAttributeNode getAttributeNode = GetAttributeNode.create();
+        @Child private UpdateShareableChildValueNode sharedAttrUpdate = UpdateShareableChildValueNode.create();
+        @Child private CastNode castStringNode;
+        private final ConditionProfile nameIsSymbolProfile = ConditionProfile.createBinaryProfile();
+
+        @Override
+        public Object executeObject(Object source, Object nameObj) {
+            String name;
+            if (nameIsSymbolProfile.profile(nameObj instanceof RSymbol)) {
+                name = ((RSymbol) nameObj).getName();
+            } else {
+                name = castToString(nameObj);
+            }
+            Object result = getAttributeNode.execute(source, name);
+            return result == null ? RNull.instance : sharedAttrUpdate.updateState(source, result);
+        }
+
+        private String castToString(Object name) {
+            if (castStringNode == null) {
+                CompilerDirectives.transferToInterpreterAndInvalidate();
+                castStringNode = insert(newCastBuilder().asStringVector().mustBe(singleElement()).findFirst().buildCastNode());
+            }
+            return (String) castStringNode.doCast(name);
+        }
+    }
+
     public abstract static class ATTRIB extends FFIUpCallNode.Arg1 {
         @Child private GetAttributesNode getAttributesNode;
 
@@ -81,14 +117,16 @@ public final class AttributesAccessNodes {
 
         @Specialization
         public Object doPairlist(RPairList obj) {
-            return obj.getTag();
+            Object result = obj.getTag();
+            assert result instanceof RSymbol || result == RNull.instance;
+            return result;
         }
 
         @Specialization
         public Object doArgs(RArgsValuesAndNames obj) {
             ArgumentsSignature signature = obj.getSignature();
             if (signature.getLength() > 0 && signature.getName(0) != null) {
-                return signature.getName(0);
+                return RDataFactory.createSymbol(signature.getName(0));
             }
             return RNull.instance;
         }
@@ -103,7 +141,7 @@ public final class AttributesAccessNodes {
                         @Cached("create()") GetNamesAttributeNode getNamesAttributeNode) {
             RStringVector names = getNamesAttributeNode.getNames(obj);
             if (names != null && names.getLength() > 0) {
-                return names.getDataAt(0);
+                return RDataFactory.createSymbol(names.getDataAt(0));
             }
             return RNull.instance;
         }
diff --git a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/CoerceNodes.java b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/CoerceNodes.java
index 63bbc904a2881dcaaa95f002b9e8672ca9e91e51..b44ce88447f0065fb2352c09368244d560fd4c21 100644
--- a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/CoerceNodes.java
+++ b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/CoerceNodes.java
@@ -123,7 +123,6 @@ public final class CoerceNodes {
     public abstract static class AsCharacterFactor extends FFIUpCallNode.Arg1 {
 
         @Child private InheritsCheckNode inheritsFactorNode = InheritsCheckNode.createFactor();
-        @Child private GetAttributeNode getAttributeNode = GetAttributeNode.create();
         @Child private RFactorNodes.GetLevels getLevels = RFactorNodes.GetLevels.create();
 
         @Specialization
diff --git a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/RfEvalNode.java b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/RfEvalNode.java
index da3d249e0dd4c25c4e487cfc76c9b701741ce7a8..ae738c977aec0e2f50ad9a5e728cbfe0f2a17a03 100644
--- a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/RfEvalNode.java
+++ b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/RfEvalNode.java
@@ -65,13 +65,13 @@ public abstract class RfEvalNode extends FFIUpCallNode.Arg2 {
     @Specialization
     @TruffleBoundary
     Object handleExpression(RExpression expr, REnvironment env) {
-        return RContext.getEngine().eval(expr, env, RCaller.topLevel);
+        return RContext.getEngine().eval(expr, env, null);
     }
 
     @Specialization
     @TruffleBoundary
     Object handleLanguage(RLanguage expr, REnvironment env) {
-        return RContext.getEngine().eval(expr, env, RCaller.topLevel);
+        return RContext.getEngine().eval(expr, env, null);
     }
 
     @Specialization
diff --git a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/TryRfEvalNode.java b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/TryRfEvalNode.java
new file mode 100644
index 0000000000000000000000000000000000000000..64d544f386ddc03116d3f24d580109a1e6988993
--- /dev/null
+++ b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/nodes/TryRfEvalNode.java
@@ -0,0 +1,44 @@
+/*
+ * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+package com.oracle.truffle.r.ffi.impl.nodes;
+
+import com.oracle.truffle.r.runtime.RErrorHandling;
+
+public class TryRfEvalNode extends FFIUpCallNode.Arg3 {
+    @Child RfEvalNode rfEvalNode = RfEvalNode.create();
+
+    @Override
+    public Object executeObject(Object expr, Object env, @SuppressWarnings("unused") Object silent) {
+        Object handlerStack = RErrorHandling.getHandlerStack();
+        Object restartStack = RErrorHandling.getRestartStack();
+        try {
+            // TODO handle silent
+            RErrorHandling.resetStacks();
+            return rfEvalNode.executeObject(expr, env);
+        } catch (Throwable t) {
+            return null;
+        } finally {
+            RErrorHandling.restoreStacks(handlerStack, restartStack);
+        }
+    }
+}
diff --git a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/upcalls/StdUpCallsRFFI.java b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/upcalls/StdUpCallsRFFI.java
index cfc4c80acb4c40e4c78fa2b1e01174778b679ab3..1f85756b25d0f1bdc1752c47bdc0b3ce200eddeb 100644
--- a/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/upcalls/StdUpCallsRFFI.java
+++ b/com.oracle.truffle.r.ffi.impl/src/com/oracle/truffle/r/ffi/impl/upcalls/StdUpCallsRFFI.java
@@ -28,6 +28,7 @@ import com.oracle.truffle.r.ffi.impl.nodes.AsLogicalNode;
 import com.oracle.truffle.r.ffi.impl.nodes.AsRealNode;
 import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodes.ATTRIB;
 import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodes.CopyMostAttrib;
+import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodes.GetAttrib;
 import com.oracle.truffle.r.ffi.impl.nodes.AttributesAccessNodes.TAG;
 import com.oracle.truffle.r.ffi.impl.nodes.CoerceNodes.CoerceVectorNode;
 import com.oracle.truffle.r.ffi.impl.nodes.CoerceNodes.VectorToPairListNode;
@@ -51,6 +52,7 @@ import com.oracle.truffle.r.ffi.impl.nodes.MiscNodes;
 import com.oracle.truffle.r.ffi.impl.nodes.MiscNodes.LENGTHNode;
 import com.oracle.truffle.r.ffi.impl.nodes.RandFunctionsNodes;
 import com.oracle.truffle.r.ffi.impl.nodes.RfEvalNode;
+import com.oracle.truffle.r.ffi.impl.nodes.TryRfEvalNode;
 import com.oracle.truffle.r.ffi.processor.RFFICstring;
 import com.oracle.truffle.r.ffi.processor.RFFIRunGC;
 import com.oracle.truffle.r.ffi.processor.RFFIUpCallNode;
@@ -122,6 +124,7 @@ public interface StdUpCallsRFFI {
     @RFFIUpCallNode(ATTRIB.class)
     Object ATTRIB(Object obj);
 
+    @RFFIUpCallNode(GetAttrib.class)
     Object Rf_getAttrib(Object obj, Object name);
 
     void Rf_setAttrib(Object obj, Object name, Object val);
@@ -271,6 +274,8 @@ public interface StdUpCallsRFFI {
 
     void Rf_copyMatrix(Object s, Object t, int byrow);
 
+    @RFFIRunGC
+    @RFFIUpCallNode(TryRfEvalNode.class)
     Object R_tryEval(Object expr, Object env, int silent);
 
     Object R_ToplevelExec();
diff --git a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/stats/model.R b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/stats/model.R
index 2175ac8d653c38340f7b99f3a51baa4a5b5bf1b8..5827c72a338b2b632e8f63746c9389a23316fe74 100644
--- a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/stats/model.R
+++ b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/stats/model.R
@@ -502,9 +502,10 @@ replaceDots <- function(x, replacement) {
         }
         return(x)
     }
-
-    for (i in seq_along(x)) {
-        x[[i]] <- replaceDots(x[[i]], replacement);
+    if (is.language(x)) {
+        for (i in seq_along(x)) {
+            x[[i]] <- replaceDots(x[[i]], replacement);
+        }
     }
     x
 }
@@ -682,7 +683,11 @@ termsform <- function (x.in, specials, data, keep.order, allowDotAsName) {
     # Step 6: Fix up the formula by substituting for dot, which should be
     # the framenames joined by +
     if (haveDot) {
-      x <- ExpandDots(x, framenames)
+        if (length(framenames) > 0) {
+            x <- ExpandDots(x, framenames)
+        } else if (!allowDotAsName) {
+           error("'.' in formula and no 'data' argument")
+        }
     }
     
     attr(x, "order") <- ord
diff --git a/com.oracle.truffle.r.native/fficall/src/truffle_common/Rinternals_truffle_common.h b/com.oracle.truffle.r.native/fficall/src/truffle_common/Rinternals_truffle_common.h
index c305453c283d0c934e99e13942a7399d78aabf1d..7a7193a66caef511c33b627cc3f3640e9d032a75 100644
--- a/com.oracle.truffle.r.native/fficall/src/truffle_common/Rinternals_truffle_common.h
+++ b/com.oracle.truffle.r.native/fficall/src/truffle_common/Rinternals_truffle_common.h
@@ -1366,7 +1366,7 @@ SEXP Rf_asS4(SEXP x, Rboolean b, int i) {
 
 static SEXP R_tryEvalInternal(SEXP x, SEXP y, int *ErrorOccurred, int silent) {
     TRACE0();
-    unimplemented("R_tryEvalInternal");
+    return ((call_R_tryEval) callbacks[R_tryEval_x])(x, y, silent);
 }
 
 SEXP R_tryEval(SEXP x, SEXP y, int *ErrorOccurred) {
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Attr.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Attr.java
index 3e2489c23cab0c3661c4ab8e883a8dc5628b0f47..7d24d0a54032ad4154071b3101f3898fdc1d55ff 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Attr.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Attr.java
@@ -110,23 +110,11 @@ public abstract class Attr extends RBuiltinNode.Arg3 {
         return container;
     }
 
-    @Specialization(guards = "!isRowNamesAttr(name)")
+    @Specialization
     protected Object attr(RAbstractContainer container, String name, boolean exact) {
         return attrRA(container, intern.execute(name), exact);
     }
 
-    @Specialization(guards = "isRowNamesAttr(name)")
-    protected Object attrRowNames(RAbstractContainer container, @SuppressWarnings("unused") String name, @SuppressWarnings("unused") boolean exact,
-                    @Cached("create()") GetRowNamesAttributeNode getRowNamesNode) {
-        // TODO: if exact == false, check for partial match (there is an ignored tests for it)
-        DynamicObject attributes = container.getAttributes();
-        if (attributes == null) {
-            return RNull.instance;
-        } else {
-            return GetAttributesNode.getFullRowNames(getRowNamesNode.getRowNames(container));
-        }
-    }
-
     /**
      * All other, non-performance centric, {@link RAttributable} types.
      */
@@ -141,8 +129,4 @@ public abstract class Attr extends RBuiltinNode.Arg3 {
             throw RError.nyi(this, "object cannot be attributed");
         }
     }
-
-    protected static boolean isRowNamesAttr(String name) {
-        return name.equals(RRuntime.ROWNAMES_ATTR_KEY);
-    }
 }
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/DynLoadFunctions.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/DynLoadFunctions.java
index d0dce21f3010b3008a6738fa038edc388dbf8e30..69aa2aa2e36212393663c077aaeb17211aa006c4 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/DynLoadFunctions.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/DynLoadFunctions.java
@@ -22,6 +22,8 @@
  */
 package com.oracle.truffle.r.nodes.builtin.base;
 
+import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.asStringVector;
+import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.instanceOf;
 import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.logicalValue;
 import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.notEmpty;
 import static com.oracle.truffle.r.nodes.builtin.CastBuilder.Predef.size;
@@ -169,14 +171,15 @@ public class DynLoadFunctions {
         static {
             Casts casts = new Casts(GetSymbolInfo.class);
             casts.arg("symbol").mustBe(stringValue()).asStringVector().mustBe(notEmpty()).findFirst();
+            casts.arg("package").defaultError(RError.Message.REQUIRES_NAME_DLLINFO).returnIf(instanceOf(RExternalPtr.class)).mustBe(stringValue()).asStringVector().findFirst();
             casts.arg("withRegistrationInfo").mustBe(logicalValue()).asLogicalVector().findFirst().map(toBoolean());
         }
 
         @Specialization
         @TruffleBoundary
-        protected Object getSymbolInfo(String symbol, String packageName, boolean withReg) {
+        protected Object getSymbolInfo(String symbol, RAbstractStringVector packageName, boolean withReg) {
             DLL.RegisteredNativeSymbol rns = DLL.RegisteredNativeSymbol.any();
-            DLL.SymbolHandle f = findSymbolNode.execute(RRuntime.asString(symbol), packageName, rns);
+            DLL.SymbolHandle f = findSymbolNode.execute(RRuntime.asString(symbol), packageName.getDataAt(0), rns);
             SymbolInfo symbolInfo = null;
             if (f != DLL.SYMBOL_NOT_FOUND) {
                 symbolInfo = new SymbolInfo(rns.getDllInfo(), symbol, f);
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Eval.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Eval.java
index 01d7eaa01723bbeef0689b6981dc96b6ef27f790..906f67906eb3c1dad6bc7903825c1530c41e04e9 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Eval.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Eval.java
@@ -48,6 +48,7 @@ import com.oracle.truffle.r.nodes.builtin.base.GetFunctionsFactory.GetNodeGen;
 import com.oracle.truffle.r.nodes.function.PromiseHelperNode.PromiseCheckHelperNode;
 import com.oracle.truffle.r.nodes.function.visibility.SetVisibilityNode;
 import com.oracle.truffle.r.runtime.ArgumentsSignature;
+import com.oracle.truffle.r.runtime.RArguments;
 import com.oracle.truffle.r.runtime.RCaller;
 import com.oracle.truffle.r.runtime.RError;
 import com.oracle.truffle.r.runtime.RType;
@@ -170,8 +171,8 @@ public abstract class Eval extends RBuiltinNode.Arg3 {
 
     @Specialization
     protected Object doEval(VirtualFrame frame, RLanguage expr, Object envir, Object enclos) {
-        RCaller rCaller = RCaller.create(frame, getOriginalCall());
         REnvironment environment = envCast.execute(frame, envir, enclos);
+        RCaller rCaller = getCaller(frame, environment);
         try {
             return RContext.getEngine().eval(expr, environment, rCaller);
         } catch (ReturnException ret) {
@@ -187,8 +188,8 @@ public abstract class Eval extends RBuiltinNode.Arg3 {
 
     @Specialization
     protected Object doEval(VirtualFrame frame, RExpression expr, Object envir, Object enclos) {
-        RCaller rCaller = RCaller.create(frame, getOriginalCall());
         REnvironment environment = envCast.execute(frame, envir, enclos);
+        RCaller rCaller = getCaller(frame, environment);
         try {
             return RContext.getEngine().eval(expr, environment, rCaller);
         } catch (ReturnException ret) {
@@ -272,4 +273,12 @@ public abstract class Eval extends RBuiltinNode.Arg3 {
         visibility.execute(frame, true);
         return expr;
     }
+
+    private RCaller getCaller(VirtualFrame frame, REnvironment environment) {
+        if (environment instanceof REnvironment.Function) {
+            return RArguments.getCall(environment.getFrame());
+        } else {
+            return RCaller.create(frame, getOriginalCall());
+        }
+    }
 }
diff --git a/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/attributes/GetAttributeNode.java b/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/attributes/GetAttributeNode.java
index dd749a55ff746724379af75f39ad64ee540aa527..ec04358a1e47b59a81508bb197cd86d2fa082433 100644
--- a/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/attributes/GetAttributeNode.java
+++ b/com.oracle.truffle.r.nodes/src/com/oracle/truffle/r/nodes/attributes/GetAttributeNode.java
@@ -32,8 +32,12 @@ import com.oracle.truffle.api.object.Shape;
 import com.oracle.truffle.api.profiles.BranchProfile;
 import com.oracle.truffle.api.profiles.ConditionProfile;
 import com.oracle.truffle.api.profiles.ValueProfile;
+import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetNamesAttributeNode;
+import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.GetRowNamesAttributeNode;
+import com.oracle.truffle.r.runtime.RRuntime;
 import com.oracle.truffle.r.runtime.data.RAttributable;
 import com.oracle.truffle.r.runtime.data.RAttributeStorage;
+import com.oracle.truffle.r.runtime.data.RNull;
 
 /**
  * This node is responsible for retrieving a value from an arbitrary attribute. It accepts both
@@ -54,8 +58,25 @@ public abstract class GetAttributeNode extends AttributeAccessNode {
 
     public abstract Object execute(Object attrs, String name);
 
+    @Specialization
+    protected RNull attr(RNull container, @SuppressWarnings("unused") String name) {
+        return container;
+    }
+
+    @Specialization(guards = "isRowNamesAttr(name)")
+    protected Object getRowNames(DynamicObject attrs, @SuppressWarnings("unused") String name,
+                    @Cached("create()") GetRowNamesAttributeNode getRowNamesNode) {
+        return GetAttributesNode.getFullRowNames(getRowNamesNode.execute(attrs));
+    }
+
+    @Specialization(guards = "isNamesAttr(name)")
+    protected Object getNames(DynamicObject attrs, @SuppressWarnings("unused") String name,
+                    @Cached("create()") GetNamesAttributeNode getNamesAttributeNode) {
+        return getNamesAttributeNode.execute(attrs);
+    }
+
     @Specialization(limit = "3", //
-                    guards = {"cachedName.equals(name)", "shapeCheck(shape, attrs)"}, //
+                    guards = {"!isSpecialAttribute(name)", "cachedName.equals(name)", "shapeCheck(shape, attrs)"}, //
                     assumptions = {"shape.getValidAssumption()"})
     protected Object getAttrCached(DynamicObject attrs, @SuppressWarnings("unused") String name,
                     @SuppressWarnings("unused") @Cached("name") String cachedName,
@@ -65,7 +86,7 @@ public abstract class GetAttributeNode extends AttributeAccessNode {
     }
 
     @TruffleBoundary
-    @Specialization(replaces = {"getAttrCached"})
+    @Specialization(replaces = "getAttrCached", guards = "!isSpecialAttribute(name)")
     protected Object getAttrFallback(DynamicObject attrs, String name) {
         return attrs.get(name);
     }
@@ -95,4 +116,16 @@ public abstract class GetAttributeNode extends AttributeAccessNode {
 
         return recursive.execute(attributes, name);
     }
+
+    protected static boolean isRowNamesAttr(String name) {
+        return name.equals(RRuntime.ROWNAMES_ATTR_KEY);
+    }
+
+    protected static boolean isNamesAttr(String name) {
+        return name.equals(RRuntime.NAMES_ATTR_KEY);
+    }
+
+    protected static boolean isSpecialAttribute(String name) {
+        return isRowNamesAttr(name) || isNamesAttr(name);
+    }
 }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/VirtualEvalFrame.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/VirtualEvalFrame.java
index 7eae87ef3866b9eb754f29a2c251b42923662118..336a2cf0b0154fd749988d7fce63335750014419 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/VirtualEvalFrame.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/VirtualEvalFrame.java
@@ -99,7 +99,12 @@ public abstract class VirtualEvalFrame extends SubstituteVirtualFrame implements
         Object[] arguments = Arrays.copyOf(originalFrame.getArguments(), originalFrame.getArguments().length);
         arguments[RArguments.INDEX_IS_IRREGULAR] = true;
         arguments[RArguments.INDEX_FUNCTION] = function;
-        arguments[RArguments.INDEX_CALL] = call;
+        if (call != null) {
+            // Otherwise leave here the call from originalFrame
+            arguments[RArguments.INDEX_CALL] = call;
+        } else if (arguments[RArguments.INDEX_CALL] == null) {
+            arguments[RArguments.INDEX_CALL] = RCaller.topLevel;
+        }
         MaterializedFrame unwrappedFrame = originalFrame instanceof SubstituteVirtualFrame ? ((SubstituteVirtualFrame) originalFrame).getOriginalFrame() : originalFrame;
         @SuppressWarnings("unchecked")
         Class<MaterializedFrame> clazz = (Class<MaterializedFrame>) unwrappedFrame.getClass();
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/context/Engine.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/context/Engine.java
index b31e710df2030d6d11415e6dfa360c7c3839b3d3..fda95437ebad45eeaccbaf9d2d422317f3ed76c4 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/context/Engine.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/context/Engine.java
@@ -199,12 +199,14 @@ public interface Engine {
     Object parseAndEval(Source sourceDesc, MaterializedFrame frame, boolean printResult) throws ParseException;
 
     /**
-     * Support for the {@code eval} {@code .Internal}.
+     * Support for the {@code eval} {@code .Internal}. If the {@code caller} argument is null, it is
+     * taken from the environment's frame.
      */
     Object eval(RExpression expr, REnvironment envir, RCaller caller);
 
     /**
      * Variant of {@link #eval(RExpression, REnvironment, RCaller)} for a single language element.
+     * If the {@code caller} argument is null, it is taken from the environment's frame.
      */
     Object eval(RLanguage expr, REnvironment envir, RCaller caller);
 
diff --git a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/R/testrffi.R b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/R/testrffi.R
index 214488fecfa13c364a9eddaf364dc30970e62758..00e256bf9356c49446f646c9ba72ec7fdd2e6328 100644
--- a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/R/testrffi.R
+++ b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/R/testrffi.R
@@ -165,6 +165,10 @@ rffi.ATTRIB <- function(x) {
     .Call('test_ATTRIB', x);
 }
 
+rffi.getAttrib <- function(source, name) {
+    .Call('test_getAttrib', source, name);
+}
+
 rffi.getStringNA <- function() {
     .Call("test_stringNA")
 }
diff --git a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.c b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.c
index 3e6edb7ee6e8f2879ee3a141e89a384faa3fab82..7ab9d8c4be721d9901d625f58f8923b70f84676b 100644
--- a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.c
+++ b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.c
@@ -347,6 +347,10 @@ SEXP test_ATTRIB(SEXP x) {
     return ATTRIB(x);
 }
 
+SEXP test_getAttrib(SEXP source, SEXP name) {
+    return Rf_getAttrib(source, name);
+}
+
 SEXP test_stringNA(void) {
     SEXP x = allocVector(STRSXP, 1);
     SET_STRING_ELT(x, 0, NA_STRING);
diff --git a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.h b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.h
index 2fbab660b916bb16973f4340025af10b5b585646..a9431321997e51f1fac2c933a7af6e5d6d25d188 100644
--- a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.h
+++ b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/src/testrffi.h
@@ -92,6 +92,8 @@ extern SEXP test_coerceVector(SEXP x, SEXP mode);
 
 extern SEXP test_ATTRIB(SEXP);
 
+extern SEXP test_getAttrib(SEXP,SEXP);
+
 extern SEXP test_stringNA(void);
 
 extern SEXP test_captureDotsWithSingleElement(SEXP env);
diff --git a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/tests/simpleTests.R b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/tests/simpleTests.R
index e1fea9c3ccf24afc755177189c3ff422497f0e28..cf85ab25b37bee4c3d962a1907f945a09187c62d 100644
--- a/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/tests/simpleTests.R
+++ b/com.oracle.truffle.r.test.native/packages/testrffi/testrffi/tests/simpleTests.R
@@ -30,6 +30,8 @@ x <- list(1)
 attr(x, 'myattr') <- 'hello';
 attrs <- rffi.ATTRIB(x)
 stopifnot(attrs[[1]] == 'hello')
+attr <- rffi.getAttrib(x, 'myattr')
+stopifnot(attr == 'hello')
 
 # loess invokes loess_raw native function passing in string value as argument and that is what we test here.
 loess(dist ~ speed, cars);
@@ -43,6 +45,21 @@ promiseInfo <- foo(tmp)
 stopifnot('some_unique_name' %in% ls(promiseInfo[[2]]))
 eval(promiseInfo[[1]], promiseInfo[[2]])
 
+# parent.frame call in Rf_eval. Simulates pattern from rlang package
+getCurrEnv <- function(r = parent.frame()) r
+fn <- function(eval_fn) {
+  list(middle(eval_fn), getCurrEnv())
+}
+middle <- function(eval_fn) {
+  deep(eval_fn, getCurrEnv())
+}
+deep <- function(eval_fn, eval_env) {
+  # the result value of rffi.tryEval is list, first element is the actual result
+  eval_fn(quote(parent.frame()), eval_env)[[1]]
+}
+res <- fn(rffi.tryEval)
+stopifnot(identical(res[[1]], res[[2]]))
+
 # fiddling the pointers to the native arrays: we get data pointer to the first SEXP argument (vec),
 # then put value 42/TRUE directly into it at index 0,
 # value of symbol 'myvar' through Rf_eval at index 1,
diff --git a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test
index 36982bf2ae2cf959a9573a433cccb1a35f50c50c..cc7e4e4fea9a0a6c3a0e75245573104b2bb8a1ae 100644
--- a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test
+++ b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/ExpectedTestOutput.test
@@ -25299,6 +25299,13 @@ x
 #{ ne <- new.env(); evalq(x <- 1, ne); ls(ne) }
 [1] "x"
 
+##com.oracle.truffle.r.test.builtins.TestBuiltin_eval.testEvalVisibility#
+#eval(parse(text='1+1'))
+[1] 2
+
+##com.oracle.truffle.r.test.builtins.TestBuiltin_eval.testEvalVisibility#
+#eval(parse(text='x<-1'))
+
 ##com.oracle.truffle.r.test.builtins.TestBuiltin_eval.testReturnInEvalExpr#
 #f1 <- function(x) { eval(quote(if(x>2){return()}else 1)); 10 };f1(5);f1(0)
 [1] 10
@@ -161126,6 +161133,30 @@ attr(,"assign")
 15 2 2 99.190
 16 2 2 16.000
 
+##com.oracle.truffle.r.test.library.stats.TestFormulae.testDotWithData#
+#terms.formula(x ~ .^4, data=data.frame(1:5))
+x ~ X1.5^4
+attr(,"variables")
+list(x, X1.5)
+attr(,"factors")
+     X1.5
+x       0
+X1.5    1
+attr(,"term.labels")
+[1] "X1.5"
+attr(,"order")
+[1] 1
+attr(,"intercept")
+[1] 1
+attr(,"response")
+[1] 1
+attr(,".Environment")
+<environment: R_GlobalEnv>
+
+##com.oracle.truffle.r.test.library.stats.TestFormulae.testDotWithNoFramenames#Output.IgnoreErrorContext#
+#terms.formula(x ~ .^4)
+Error in terms.formula(x ~ .^4) : '.' in formula and no 'data' argument
+
 ##com.oracle.truffle.r.test.library.stats.TestFormulae.testExpandDostsAndSpecialsTermsform#
 #f <- terms.formula(cyl~myfun(mpg)+., specials=c('myfun'), data=mtcars); attrs <- attributes(f); envIdx <- which(names(attrs)=='.Environment'); print(attrs[envIdx]); attrs[sort(names(attrs[-envIdx]))]
 $.Environment
diff --git a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_eval.java b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_eval.java
index 55adc7d42e2b2566e2abe645d2fab23070a85a6c..fc2a2d8923fec2e133c1af559ba429a42d54e5e9 100644
--- a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_eval.java
+++ b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/builtins/TestBuiltin_eval.java
@@ -68,4 +68,10 @@ public class TestBuiltin_eval extends TestBase {
     public void testReturnInEvalExpr() {
         assertEval("f1 <- function(x) { eval(quote(if(x>2){return()}else 1)); 10 };f1(5);f1(0)");
     }
+
+    @Test
+    public void testEvalVisibility() {
+        assertEval("eval(parse(text='x<-1'))");
+        assertEval("eval(parse(text='1+1'))");
+    }
 }
diff --git a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/library/stats/TestFormulae.java b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/library/stats/TestFormulae.java
index 76faa10ddb6f317796b42ff353e03a8aee98c84d..06692c4ac63f82f90e12a215374721b6742e7037 100644
--- a/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/library/stats/TestFormulae.java
+++ b/com.oracle.truffle.r.test/src/com/oracle/truffle/r/test/library/stats/TestFormulae.java
@@ -105,4 +105,14 @@ public class TestFormulae extends TestBase {
     public void testExpandDostsAndSpecialsTermsform() {
         assertEval("f <- terms.formula(cyl~myfun(mpg)+., specials=c('myfun'), data=mtcars); attrs <- attributes(f); " + SHOW_ATTRS);
     }
+
+    @Test
+    public void testDotWithData() {
+        assertEval("terms.formula(x ~ .^4, data=data.frame(1:5))");
+    }
+
+    @Test
+    public void testDotWithNoFramenames() {
+        assertEval(Output.IgnoreErrorContext, "terms.formula(x ~ .^4)");
+    }
 }