예제 #1
0
        public static string to_numpy_string(Tensor tensor)
        {
            var dtype = tensor.dtype;

            if (dtype == TF_DataType.TF_STRING && tensor.NDims > 0)
            {
                return($"['{string.Join("', '", tensor.StringData())}']");
            }

            var nd = tensor.numpy();

            if (nd.size == 0)
            {
                return("[]");
            }

            switch (dtype)
            {
            case TF_DataType.TF_STRING:
                return(string.Join(string.Empty, nd.ToArray <byte>()
                                   .Select(x => x < 32 || x > 127 ? "\\x" + x.ToString("x") : Convert.ToChar(x).ToString())));

            case TF_DataType.TF_BOOL:
                return((nd.GetByte(0) > 0).ToString());

            case TF_DataType.TF_VARIANT:
            case TF_DataType.TF_RESOURCE:
                return("<unprintable>");

            default:
                return(nd.ToString());
            }
        }
예제 #2
0
        /// <summary>
        /// Helper function for reduction ops.
        /// </summary>
        /// <param name="input_shape">1-D Tensor, the shape of the Tensor being reduced.</param>
        /// <param name="axes">1-D Tensor, the reduction axes.</param>
        /// <returns>A 1-D Tensor, the output shape as if keepdims were set to True.</returns>
        public static Tensor reduced_shape(Tensor input_shape, Tensor axes)
        {
            if (tf.Context.executing_eagerly())
            {
                var input_shape_val = input_shape.numpy();
                foreach (var axes_val in axes.numpy().ToArray <int>())
                {
                    input_shape_val[axes_val] = 1;
                }
                return(tf.constant(input_shape_val));
            }

            input_shape = to_int32(input_shape);
            axes        = to_int32(axes);

            var input_rank = array_ops.size(input_shape);

            axes = (axes + input_rank) % input_rank;
            var axes_shape = array_ops.shape(axes);
            var rng        = math_ops.range(input_rank);
            var a1         = new Tensor[] { rng, axes };
            var fill       = gen_array_ops.fill(axes_shape, 1);
            var a2         = new Tensor[] { input_shape, fill };

            return(gen_data_flow_ops.dynamic_stitch(a1, a2));
        }
예제 #3
0
        public static string to_numpy_string(Tensor tensor)
        {
            var dtype = tensor.dtype;

            if (dtype == TF_DataType.TF_STRING)
            {
                if (tensor.rank == 0)
                {
                    return("'" + string.Join(string.Empty, tensor.StringBytes()[0]
                                             .Take(25)
                                             .Select(x => x < 32 || x > 127 ? "\\x" + x.ToString("x") : Convert.ToChar(x).ToString())) + "'");
                }
                else
                {
                    return($"['{string.Join("', '", tensor.StringData().Take(25))}']");
                }
            }
            else if (dtype == TF_DataType.TF_VARIANT)
            {
                return("<unprintable>");
            }
            else if (dtype == TF_DataType.TF_RESOURCE)
            {
                return("<unprintable>");
            }

            var nd = tensor.numpy();

            if (nd.size == 0)
            {
                return("[]");
            }

            return(nd.ToString());
        }
예제 #4
0
        /// <summary>
        /// Returns the constant value of the given tensor, if efficiently calculable.
        /// </summary>
        /// <param name="tensor"></param>
        /// <param name="partial"></param>
        /// <returns></returns>
        public static NDArray constant_value(Tensor tensor, bool partial = false)
        {
            if (tensor is EagerTensor)
            {
                return(tensor.numpy());
            }

            NDArray ret = _ConstantValue(tensor, partial);

            if (!(ret is null))
            {
                tensor.graph.prevent_feeding(tensor);
            }

            return(ret);
        }
예제 #5
0
        public static TensorShape constant_value_as_shape(Tensor tensor)
        {
            bool hasattr(Graph property, string attr)
            {
                var t = property.GetType().GetProperties();

                foreach (System.Reflection.PropertyInfo pi in t)
                {
                    if (pi.Name == attr)
                    {
                        return(true);
                    }
                }
                return(false);
            }

            if (tensor.GetType() == typeof(EagerTensor))
            {
                return(new TensorShape(tensor.numpy().ToArray <int>()));
            }

            if (tensor.TensorShape.ndim == 0)
            {
                var value_ = constant_value(tensor);
                if (value_ == null)
                {
                    throw new ValueError(
                              @"Received a scalar with unknown value as shape; require a statically
known scalar with value '-1' to describe an unknown shape.");
                }
                if (value_ != -1)
                {
                    throw new ValueError(
                              String.Format(@"Received a scalar value {0} as shape; require a statically known
scalar with value '-1' to describe an unknown shape.", value_));
                }
                return(tensor.TensorShape.unknown_shape(-1));
            }

            var shape = tensor.TensorShape.with_rank(1);

            if (shape == new TensorShape(new int[] { 1 }))
            {
                return(new TensorShape(new int[] { }));
            }
            else if (tensor.op.type == "Cast")
            {
                var pre_cast = constant_value_as_shape(tensor.op.inputs[0]);
                if (pre_cast.dims == null)
                {
                    return(pre_cast);
                }
                var cast_dtype = dtypes.as_dtype((Type)tensor.op.get_attr("DstT"));
                if (!Array.Exists(new[] { dtypes.int32, dtypes.int64 }, cast_dtype_ => cast_dtype_ == cast_dtype))
                {
                    return(tensor.TensorShape.unknown_shape(shape.dims[0]));
                }

                int[] x_ = { };
                foreach (var x in pre_cast.as_list())
                {
                    if (x != -1)
                    {
                        x_[x_.Length] = x;
                    }
                    else
                    {
                        x_[x_.Length] = -1;
                    }
                }
                var dest_dtype_shape_array = np.array(x_).astype(cast_dtype.as_numpy_dtype());

                int[] y_ = { };
                foreach (int y in dest_dtype_shape_array)
                {
                    if (y >= 0)
                    {
                        y_[y_.Length] = y;
                    }
                    else
                    {
                        y_[y_.Length] = -1;
                    }
                }
                return(new TensorShape(y_));
            }
            else if (tensor.op.type == "Shape")
            {
                return(tensor.op.inputs[0].shape);
            }
            else if (tensor.op.type == "Pack")
            {
                var ret_ = new TensorShape(new int[] { });
                if ((int)tensor.op.get_attr("axis") != 0)
                {
                    throw new ValueError(String.Format(
                                             @"Since rank 1 inputs are expected, Pack's axis: {0} must be 0, otherwise it
would not be rank 1.", tensor.op.get_attr("axis")));
                }
                foreach (Tensor pack_input in tensor.op.inputs)
                {
                    var       pack_input_val = constant_value(pack_input);
                    Dimension new_dim;
                    if (pack_input_val < 0)
                    {
                        new_dim = new Dimension(-1);
                    }
                    else if (pack_input_val == null)
                    {
                        new_dim = new Dimension(-1);
                    }
                    else
                    {
                        new_dim = new Dimension(pack_input_val);
                    }
                    ret_ = ret_.concatenate(new int[] { new_dim });
                }
                return(ret_);
            }
            else if (tensor.op.type == "Concat")
            {
                var ret_ = new TensorShape(new int[] { });

                var inputlist_ = new ArraySegment <Tensor>(tensor.op.inputs, 1,
                                                           tensor.op.inputs.Length - 1);
                foreach (var concat_input in inputlist_)
                {
                    ret_ = ret_.concatenate(constant_value_as_shape(concat_input));
                }
                return(ret_);
            }
            else if (tensor.op.type == "StridedSlice")
            {
                try
                {
                    var begin   = constant_value(tensor.op.inputs[1]);
                    var end     = constant_value(tensor.op.inputs[2]);
                    var strides = constant_value(tensor.op.inputs[3]);
                    if (new[] { begin, end, strides }.All(x => x == null))
                    {
                        begin   = begin[0];
                        end     = end[0];
                        strides = strides[0];
                        var begin_mask = tensor.op.get_attr("begin_mask");
                        if ((int)begin_mask == 1)
                        {
                            begin = null;
                        }
                        var end_mask = tensor.op.get_attr("end_mask");
                        if ((int)end_mask == 1)
                        {
                            end = null;
                        }

                        var ellipsis_mask    = tensor.op.get_attr("ellipsis_mask");
                        var new_axis_mask    = tensor.op.get_attr("new_axis_mask");
                        var shrink_axis_mask = tensor.op.get_attr("shrink_axis_mask");

                        bool valid_attributes;
                        if (!(bool)ellipsis_mask && !(bool)new_axis_mask &&
                            !(bool)shrink_axis_mask && !((bool)begin_mask || (int)begin_mask == 1) &&
                            !((bool)end_mask || (int)end_mask == 1))
                        {
                            valid_attributes = true;
                        }
                        else
                        {
                            valid_attributes = false;
                        }
                        if (valid_attributes)
                        {
                            // sorry for the mess here, but this hacky solution was the best way
                            // i could come up with to implement the things done in python in c#
                            var prev_ = constant_value_as_shape(tensor.op.inputs[0]).dims;
                            var prev  = prev_.Skip(begin).Take(end - begin).ToArray();
                            // 100 being the comparison doesn't really matter here; it's going to break anyway
                            for (int iter = 0; iter != 100; iter = iter + strides)
                            {
                                prev[prev.Length] = prev_[iter];
                                if ((iter + strides) > prev_.Length)
                                {
                                    break;
                                }
                            }
                            var ret_ = new TensorShape(prev);
                            return(ret_);
                        }
                    }
                }
                catch (Exception ex)
                {
                    if (ex is ValueError || ex is TypeError)
                    {
                    }
                }
            }
            else if (tensor.op.type == "Placeholder" &&
                     tensor.op.graph.building_function &&
                     tensor.op.graph is FuncGraph func_graph)
            {
                int i = 0;
                foreach (Tensor capture in func_graph.internal_captures)
                {
                    if (capture.GetType() == typeof(Tensor))
                    {
                        var external_capture = func_graph.external_captures[i];
                        return(constant_value_as_shape(external_capture));
                    }

                    i++;
                }
            }

            var ret   = tensor.TensorShape.unknown_shape(shape.dims[0]);
            var value = constant_value(tensor);

            if (!(value is null))
            {
                var d_ = new int[value.size];
                foreach (var(index, d) in enumerate(value.ToArray <int>()))
                {
                    d_[index] = d >= 0 ? d : -1;
                }

                ret = ret.merge_with(new TensorShape(d_));
            }
            return(ret);
        }