/**
|
* @license
|
* Copyright 2017 Google LLC. All Rights Reserved.
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
* =============================================================================
|
*/
|
// Workaround for: https://github.com/bazelbuild/rules_nodejs/issues/1265
|
/// <reference types="@webgpu/types/dist" />
|
import { getGlobal } from './global_util';
|
import { tensorToString } from './tensor_format';
|
import * as util from './util';
|
import { computeStrides, toNestedArray } from './util';
|
/**
|
* A mutable object, similar to `tf.Tensor`, that allows users to set values
|
* at locations before converting to an immutable `tf.Tensor`.
|
*
|
* See `tf.buffer` for creating a tensor buffer.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
export class TensorBuffer {
|
constructor(shape, dtype, values) {
|
this.dtype = dtype;
|
this.shape = shape.slice();
|
this.size = util.sizeFromShape(shape);
|
if (values != null) {
|
const n = values.length;
|
util.assert(n === this.size, () => `Length of values '${n}' does not match the size ` +
|
`inferred by the shape '${this.size}'.`);
|
}
|
if (dtype === 'complex64') {
|
throw new Error(`complex64 dtype TensorBuffers are not supported. Please create ` +
|
`a TensorBuffer for the real and imaginary parts separately and ` +
|
`call tf.complex(real, imag).`);
|
}
|
this.values = values || util.getArrayFromDType(dtype, this.size);
|
this.strides = computeStrides(shape);
|
}
|
/**
|
* Sets a value in the buffer at a given location.
|
*
|
* @param value The value to set.
|
* @param locs The location indices.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Creation'}
|
*/
|
set(value, ...locs) {
|
if (locs.length === 0) {
|
locs = [0];
|
}
|
util.assert(locs.length === this.rank, () => `The number of provided coordinates (${locs.length}) must ` +
|
`match the rank (${this.rank})`);
|
const index = this.locToIndex(locs);
|
this.values[index] = value;
|
}
|
/**
|
* Returns the value in the buffer at the provided location.
|
*
|
* @param locs The location indices.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Creation'}
|
*/
|
get(...locs) {
|
if (locs.length === 0) {
|
locs = [0];
|
}
|
let i = 0;
|
for (const loc of locs) {
|
if (loc < 0 || loc >= this.shape[i]) {
|
const msg = `Requested out of range element at ${locs}. ` +
|
` Buffer shape=${this.shape}`;
|
throw new Error(msg);
|
}
|
i++;
|
}
|
let index = locs[locs.length - 1];
|
for (let i = 0; i < locs.length - 1; ++i) {
|
index += this.strides[i] * locs[i];
|
}
|
return this.values[index];
|
}
|
locToIndex(locs) {
|
if (this.rank === 0) {
|
return 0;
|
}
|
else if (this.rank === 1) {
|
return locs[0];
|
}
|
let index = locs[locs.length - 1];
|
for (let i = 0; i < locs.length - 1; ++i) {
|
index += this.strides[i] * locs[i];
|
}
|
return index;
|
}
|
indexToLoc(index) {
|
if (this.rank === 0) {
|
return [];
|
}
|
else if (this.rank === 1) {
|
return [index];
|
}
|
const locs = new Array(this.shape.length);
|
for (let i = 0; i < locs.length - 1; ++i) {
|
locs[i] = Math.floor(index / this.strides[i]);
|
index -= locs[i] * this.strides[i];
|
}
|
locs[locs.length - 1] = index;
|
return locs;
|
}
|
get rank() {
|
return this.shape.length;
|
}
|
/**
|
* Creates an immutable `tf.Tensor` object from the buffer.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Creation'}
|
*/
|
toTensor() {
|
return trackerFn().makeTensor(this.values, this.shape, this.dtype);
|
}
|
}
|
// For tracking tensor creation and disposal.
|
let trackerFn = null;
|
// Used by chaining methods to call into ops.
|
let opHandler = null;
|
// Used to warn about deprecated methods.
|
let deprecationWarningFn = null;
|
// This here so that we can use this method on dev branches and keep the
|
// functionality at master.
|
// tslint:disable-next-line:no-unused-expression
|
[deprecationWarningFn];
|
/**
|
* An external consumer can register itself as the tensor tracker. This way
|
* the Tensor class can notify the tracker for every tensor created and
|
* disposed.
|
*/
|
export function setTensorTracker(fn) {
|
trackerFn = fn;
|
}
|
/**
|
* An external consumer can register itself as the op handler. This way the
|
* Tensor class can have chaining methods that call into ops via the op
|
* handler.
|
*/
|
export function setOpHandler(handler) {
|
opHandler = handler;
|
}
|
/**
|
* Sets the deprecation warning function to be used by this file. This way the
|
* Tensor class can be a leaf but still use the environment.
|
*/
|
export function setDeprecationWarningFn(fn) {
|
deprecationWarningFn = fn;
|
}
|
/**
|
* A `tf.Tensor` object represents an immutable, multidimensional array of
|
* numbers that has a shape and a data type.
|
*
|
* For performance reasons, functions that create tensors do not necessarily
|
* perform a copy of the data passed to them (e.g. if the data is passed as a
|
* `Float32Array`), and changes to the data will change the tensor. This is not
|
* a feature and is not supported. To avoid this behavior, use the tensor before
|
* changing the input data or create a copy with `copy = tf.add(yourTensor, 0)`.
|
*
|
* See `tf.tensor` for details on how to create a `tf.Tensor`.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
export class Tensor {
|
constructor(shape, dtype, dataId, id) {
|
/** Whether this tensor has been globally kept. */
|
this.kept = false;
|
this.isDisposedInternal = false;
|
this.shape = shape.slice();
|
this.dtype = dtype || 'float32';
|
this.size = util.sizeFromShape(shape);
|
this.strides = computeStrides(shape);
|
this.dataId = dataId;
|
this.id = id;
|
this.rankType = (this.rank < 5 ? this.rank.toString() : 'higher');
|
}
|
get rank() {
|
return this.shape.length;
|
}
|
/**
|
* Returns a promise of `tf.TensorBuffer` that holds the underlying data.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
async buffer() {
|
const vals = await this.data();
|
return opHandler.buffer(this.shape, this.dtype, vals);
|
}
|
/**
|
* Returns a `tf.TensorBuffer` that holds the underlying data.
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
bufferSync() {
|
return opHandler.buffer(this.shape, this.dtype, this.dataSync());
|
}
|
/**
|
* Returns the tensor data as a nested array. The transfer of data is done
|
* asynchronously.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
async array() {
|
const vals = await this.data();
|
return toNestedArray(this.shape, vals, this.dtype === 'complex64');
|
}
|
/**
|
* Returns the tensor data as a nested array. The transfer of data is done
|
* synchronously.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
arraySync() {
|
return toNestedArray(this.shape, this.dataSync(), this.dtype === 'complex64');
|
}
|
/**
|
* Asynchronously downloads the values from the `tf.Tensor`. Returns a
|
* promise of `TypedArray` that resolves when the computation has finished.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
async data() {
|
this.throwIfDisposed();
|
const data = trackerFn().read(this.dataId);
|
if (this.dtype === 'string') {
|
const bytes = await data;
|
try {
|
return bytes.map(b => util.decodeString(b));
|
}
|
catch (_a) {
|
throw new Error('Failed to decode the string bytes into utf-8. ' +
|
'To get the original bytes, call tensor.bytes().');
|
}
|
}
|
return data;
|
}
|
/**
|
* Copy the tensor's data to a new GPU resource. Comparing to the `dataSync()`
|
* and `data()`, this method prevents data from being downloaded to CPU.
|
*
|
* For WebGL backend, the data will be stored on a densely packed texture.
|
* This means that the texture will use the RGBA channels to store value.
|
*
|
* For WebGPU backend, the data will be stored on a buffer. There is no
|
* parameter, so can not use a user-defined size to create the buffer.
|
*
|
* @param options:
|
* For WebGL,
|
* - customTexShape: Optional. If set, will use the user defined
|
* texture shape to create the texture.
|
*
|
* @returns For WebGL backend, a GPUData contains the new texture and
|
* its information.
|
* {
|
* tensorRef: The tensor that is associated with this texture,
|
* texture: WebGLTexture,
|
* texShape: [number, number] // [height, width]
|
* }
|
*
|
* For WebGPU backend, a GPUData contains the new buffer.
|
* {
|
* tensorRef: The tensor that is associated with this buffer,
|
* buffer: GPUBuffer,
|
* }
|
*
|
* Remember to dispose the GPUData after it is used by
|
* `res.tensorRef.dispose()`.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
dataToGPU(options) {
|
this.throwIfDisposed();
|
return trackerFn().readToGPU(this.dataId, options);
|
}
|
/**
|
* Synchronously downloads the values from the `tf.Tensor`. This blocks the
|
* UI thread until the values are ready, which can cause performance issues.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
dataSync() {
|
this.throwIfDisposed();
|
const data = trackerFn().readSync(this.dataId);
|
if (this.dtype === 'string') {
|
try {
|
return data.map(b => util.decodeString(b));
|
}
|
catch (_a) {
|
throw new Error('Failed to decode the string bytes into utf-8. ' +
|
'To get the original bytes, call tensor.bytes().');
|
}
|
}
|
return data;
|
}
|
/** Returns the underlying bytes of the tensor's data. */
|
async bytes() {
|
this.throwIfDisposed();
|
const data = await trackerFn().read(this.dataId);
|
if (this.dtype === 'string') {
|
return data;
|
}
|
else {
|
return new Uint8Array(data.buffer);
|
}
|
}
|
/**
|
* Disposes `tf.Tensor` from memory.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
dispose() {
|
if (this.isDisposed) {
|
return;
|
}
|
if (this.kerasMask) {
|
this.kerasMask.dispose();
|
}
|
trackerFn().disposeTensor(this);
|
this.isDisposedInternal = true;
|
}
|
get isDisposed() {
|
return this.isDisposedInternal;
|
}
|
throwIfDisposed() {
|
if (this.isDisposed) {
|
throw new Error(`Tensor is disposed.`);
|
}
|
}
|
/**
|
* Prints the `tf.Tensor`. See `tf.print` for details.
|
*
|
* @param verbose Whether to print verbose information about the tensor,
|
* including dtype and size.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
print(verbose = false) {
|
return opHandler.print(this, verbose);
|
}
|
/**
|
* Returns a copy of the tensor. See `tf.clone` for details.
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
clone() {
|
this.throwIfDisposed();
|
return opHandler.clone(this);
|
}
|
/**
|
* Returns a human-readable description of the tensor. Useful for logging.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
toString(verbose = false) {
|
const vals = this.dataSync();
|
return tensorToString(vals, this.shape, this.dtype, verbose);
|
}
|
cast(dtype) {
|
this.throwIfDisposed();
|
return opHandler.cast(this, dtype);
|
}
|
variable(trainable = true, name, dtype) {
|
this.throwIfDisposed();
|
return trackerFn().makeVariable(this, trainable, name, dtype);
|
}
|
}
|
Object.defineProperty(Tensor, Symbol.hasInstance, {
|
value: (instance) => {
|
// Implementation note: we should use properties of the object that will be
|
// defined before the constructor body has finished executing (methods).
|
// This is because when this code is transpiled by babel, babel will call
|
// classCallCheck before the constructor body is run.
|
// See https://github.com/tensorflow/tfjs/issues/3384 for backstory.
|
return !!instance && instance.data != null && instance.dataSync != null &&
|
instance.throwIfDisposed != null;
|
}
|
});
|
export function getGlobalTensorClass() {
|
// Use getGlobal so that we can augment the Tensor class across package
|
// boundaries becase the node resolution alg may result in different modules
|
// being returned for this file depending on the path they are loaded from.
|
return getGlobal('Tensor', () => {
|
return Tensor;
|
});
|
}
|
// Global side effect. Cache global reference to Tensor class
|
getGlobalTensorClass();
|
/**
|
* A mutable `tf.Tensor`, useful for persisting state, e.g. for training.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
export class Variable extends Tensor {
|
constructor(initialValue, trainable, name, tensorId) {
|
super(initialValue.shape, initialValue.dtype, initialValue.dataId, tensorId);
|
this.trainable = trainable;
|
this.name = name;
|
}
|
/**
|
* Assign a new `tf.Tensor` to this variable. The new `tf.Tensor` must have
|
* the same shape and dtype as the old `tf.Tensor`.
|
*
|
* @param newValue New tensor to be assigned to this variable.
|
*
|
* @doc {heading: 'Tensors', subheading: 'Classes'}
|
*/
|
assign(newValue) {
|
if (newValue.dtype !== this.dtype) {
|
throw new Error(`dtype of the new value (${newValue.dtype}) and ` +
|
`previous value (${this.dtype}) must match`);
|
}
|
if (!util.arraysEqual(newValue.shape, this.shape)) {
|
throw new Error(`shape of the new value (${newValue.shape}) and ` +
|
`previous value (${this.shape}) must match`);
|
}
|
trackerFn().disposeTensor(this);
|
this.dataId = newValue.dataId;
|
trackerFn().incRef(this, null /* backend */);
|
}
|
dispose() {
|
trackerFn().disposeVariable(this);
|
this.isDisposedInternal = true;
|
}
|
}
|
Object.defineProperty(Variable, Symbol.hasInstance, {
|
value: (instance) => {
|
return instance instanceof Tensor && instance.assign != null &&
|
instance.assign instanceof Function;
|
}
|
});
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tensor.js","sourceRoot":"","sources":["../../../../../tfjs-core/src/tensor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright 2017 Google LLC. All Rights Reserved.\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n * =============================================================================\n */\n\n// Workaround for: https://github.com/bazelbuild/rules_nodejs/issues/1265\n/// <reference types=\"@webgpu/types/dist\" />\n\nimport {getGlobal} from './global_util';\nimport {tensorToString} from './tensor_format';\nimport {DataId, TensorInfo} from './tensor_info';\nimport {ArrayMap, BackendValues, DataType, DataTypeMap, DataValues, NumericDataType, Rank, ShapeMap, SingleValueMap, TypedArray} from './types';\nimport * as util from './util';\nimport {computeStrides, toNestedArray} from './util';\n\nexport interface TensorData<D extends DataType> {\n  dataId?: DataId;\n  values?: DataTypeMap[D];\n}\n\n// This interface mimics KernelBackend (in backend.ts), which would create a\n// circular dependency if imported.\nexport interface Backend {}\n\n/**\n * A mutable object, similar to `tf.Tensor`, that allows users to set values\n * at locations before converting to an immutable `tf.Tensor`.\n *\n * See `tf.buffer` for creating a tensor buffer.\n *\n * @doc {heading: 'Tensors', subheading: 'Classes'}\n */\nexport class TensorBuffer<R extends Rank, D extends DataType = 'float32'> {\n  size: number;\n  shape: ShapeMap[R];\n  strides: number[];\n  values: DataTypeMap[D];\n\n  constructor(shape: ShapeMap[R], public dtype: D, values?: DataTypeMap[D]) {\n    this.shape = shape.slice() as ShapeMap[R];\n    this.size = util.sizeFromShape(shape);\n\n    if (values != null) {\n      const n = values.length;\n      util.assert(\n          n === this.size,\n          () => `Length of values '${n}' does not match the size ` +\n              `inferred by the shape '${this.size}'.`);\n    }\n    if (dtype === 'complex64') {\n      throw new Error(\n          `complex64 dtype TensorBuffers are not supported. Please create ` +\n          `a TensorBuffer for the real and imaginary parts separately and ` +\n          `call tf.complex(real, imag).`);\n    }\n    this.values = values || util.getArrayFromDType(dtype, this.size);\n    this.strides = computeStrides(shape);\n  }\n\n  /**\n   * Sets a value in the buffer at a given location.\n   *\n   * @param value The value to set.\n   * @param locs  The location indices.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Creation'}\n   */\n  set(value: SingleValueMap[D], ...locs: number[]): void {\n    if (locs.length === 0) {\n      locs = [0];\n    }\n    util.assert(\n        locs.length === this.rank,\n        () => `The number of provided coordinates (${locs.length}) must ` +\n            `match the rank (${this.rank})`);\n\n    const index = this.locToIndex(locs);\n    this.values[index] = value as number;\n  }\n\n  /**\n   * Returns the value in the buffer at the provided location.\n   *\n   * @param locs The location indices.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Creation'}\n   */\n  get(...locs: number[]): SingleValueMap[D] {\n    if (locs.length === 0) {\n      locs = [0];\n    }\n    let i = 0;\n    for (const loc of locs) {\n      if (loc < 0 || loc >= this.shape[i]) {\n        const msg = `Requested out of range element at ${locs}. ` +\n            `  Buffer shape=${this.shape}`;\n        throw new Error(msg);\n      }\n      i++;\n    }\n    let index = locs[locs.length - 1];\n    for (let i = 0; i < locs.length - 1; ++i) {\n      index += this.strides[i] * locs[i];\n    }\n    return this.values[index] as SingleValueMap[D];\n  }\n\n  locToIndex(locs: number[]): number {\n    if (this.rank === 0) {\n      return 0;\n    } else if (this.rank === 1) {\n      return locs[0];\n    }\n    let index = locs[locs.length - 1];\n    for (let i = 0; i < locs.length - 1; ++i) {\n      index += this.strides[i] * locs[i];\n    }\n    return index;\n  }\n\n  indexToLoc(index: number): number[] {\n    if (this.rank === 0) {\n      return [];\n    } else if (this.rank === 1) {\n      return [index];\n    }\n    const locs: number[] = new Array(this.shape.length);\n    for (let i = 0; i < locs.length - 1; ++i) {\n      locs[i] = Math.floor(index / this.strides[i]);\n      index -= locs[i] * this.strides[i];\n    }\n    locs[locs.length - 1] = index;\n    return locs;\n  }\n\n  get rank() {\n    return this.shape.length;\n  }\n\n  /**\n   * Creates an immutable `tf.Tensor` object from the buffer.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Creation'}\n   */\n  toTensor(): Tensor<R> {\n    return trackerFn().makeTensor(this.values, this.shape, this.dtype) as\n        Tensor<R>;\n  }\n}\n\nexport interface DataToGPUWebGLOption {\n  customTexShape?: [number, number];\n}\n\nexport type DataToGPUOptions = DataToGPUWebGLOption;\n\nexport interface GPUData {\n  tensorRef: Tensor;\n  texture?: WebGLTexture;\n  buffer?: GPUBuffer;\n  texShape?: [number, number];\n}\n\nexport interface TensorTracker {\n  makeTensor(\n      values: DataValues, shape: number[], dtype: DataType,\n      backend?: Backend): Tensor;\n  makeVariable(\n      initialValue: Tensor, trainable?: boolean, name?: string,\n      dtype?: DataType): Variable;\n  incRef(a: Tensor, backend: Backend): void;\n  disposeTensor(t: Tensor): void;\n  disposeVariable(v: Variable): void;\n  read(dataId: DataId): Promise<BackendValues>;\n  readSync(dataId: DataId): BackendValues;\n  readToGPU(dataId: DataId, options?: DataToGPUOptions): GPUData;\n}\n\n/**\n * The Tensor class calls into this handler to delegate chaining operations.\n */\nexport interface OpHandler {\n  cast<T extends Tensor>(x: T, dtype: DataType): T;\n  buffer<R extends Rank, D extends DataType>(\n      shape: ShapeMap[R], dtype: D,\n      values?: DataTypeMap[D]): TensorBuffer<R, D>;\n  print<T extends Tensor>(x: T, verbose: boolean): void;\n  clone<T extends Tensor>(x: T): T;\n  // TODO(yassogba) bring reshape back?\n}\n\n// For tracking tensor creation and disposal.\nlet trackerFn: () => TensorTracker = null;\n// Used by chaining methods to call into ops.\nlet opHandler: OpHandler = null;\n// Used to warn about deprecated methods.\nlet deprecationWarningFn: (msg: string) => void = null;\n// This here so that we can use this method on dev branches and keep the\n// functionality at master.\n// tslint:disable-next-line:no-unused-expression\n[deprecationWarningFn];\n\n/**\n * An external consumer can register itself as the tensor tracker. This way\n * the Tensor class can notify the tracker for every tensor created and\n * disposed.\n */\nexport function setTensorTracker(fn: () => TensorTracker) {\n  trackerFn = fn;\n}\n\n/**\n * An external consumer can register itself as the op handler. This way the\n * Tensor class can have chaining methods that call into ops via the op\n * handler.\n */\nexport function setOpHandler(handler: OpHandler) {\n  opHandler = handler;\n}\n\n/**\n * Sets the deprecation warning function to be used by this file. This way the\n * Tensor class can be a leaf but still use the environment.\n */\nexport function setDeprecationWarningFn(fn: (msg: string) => void) {\n  deprecationWarningFn = fn;\n}\n\n// Declare this namespace to make Tensor class augmentation work in google3.\nexport declare namespace Tensor {}\n/**\n * A `tf.Tensor` object represents an immutable, multidimensional array of\n * numbers that has a shape and a data type.\n *\n * For performance reasons, functions that create tensors do not necessarily\n * perform a copy of the data passed to them (e.g. if the data is passed as a\n * `Float32Array`), and changes to the data will change the tensor. This is not\n * a feature and is not supported. To avoid this behavior, use the tensor before\n * changing the input data or create a copy with `copy = tf.add(yourTensor, 0)`.\n *\n * See `tf.tensor` for details on how to create a `tf.Tensor`.\n *\n * @doc {heading: 'Tensors', subheading: 'Classes'}\n */\nexport class Tensor<R extends Rank = Rank> implements TensorInfo {\n  /** Unique id of this tensor. */\n  readonly id: number;\n  /**\n   * Id of the bucket holding the data for this tensor. Multiple arrays can\n   * point to the same bucket (e.g. when calling array.reshape()).\n   */\n  dataId: DataId;\n  /** The shape of the tensor. */\n  readonly shape: ShapeMap[R];\n  /** Number of elements in the tensor. */\n  readonly size: number;\n  /** The data type for the array. */\n  readonly dtype: DataType;\n  /** The rank type for the array (see `Rank` enum). */\n  readonly rankType: R;\n\n  /** Whether this tensor has been globally kept. */\n  kept = false;\n  /** The id of the scope this tensor is being tracked in. */\n  scopeId: number;\n  /** The keras mask that some keras layers attach to the tensor */\n  kerasMask?: Tensor;\n\n  /**\n   * Number of elements to skip in each dimension when indexing. See\n   * https://docs.scipy.org/doc/numpy/reference/generated/\\\n   * numpy.ndarray.strides.html\n   */\n  readonly strides: number[];\n\n  constructor(shape: ShapeMap[R], dtype: DataType, dataId: DataId, id: number) {\n    this.shape = shape.slice() as ShapeMap[R];\n    this.dtype = dtype || 'float32';\n    this.size = util.sizeFromShape(shape);\n    this.strides = computeStrides(shape);\n    this.dataId = dataId;\n    this.id = id;\n    this.rankType = (this.rank < 5 ? this.rank.toString() : 'higher') as R;\n  }\n\n  get rank(): number {\n    return this.shape.length;\n  }\n\n  /**\n   * Returns a promise of `tf.TensorBuffer` that holds the underlying data.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  async buffer<D extends DataType = 'float32'>(): Promise<TensorBuffer<R, D>> {\n    const vals = await this.data<D>();\n    return opHandler.buffer(this.shape, this.dtype as D, vals);\n  }\n\n  /**\n   * Returns a `tf.TensorBuffer` that holds the underlying data.\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  bufferSync<D extends DataType = 'float32'>(): TensorBuffer<R, D> {\n    return opHandler.buffer(this.shape, this.dtype as D, this.dataSync());\n  }\n\n  /**\n   * Returns the tensor data as a nested array. The transfer of data is done\n   * asynchronously.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  async array(): Promise<ArrayMap[R]> {\n    const vals = await this.data();\n    return toNestedArray(this.shape, vals, this.dtype === 'complex64') as\n        ArrayMap[R];\n  }\n\n  /**\n   * Returns the tensor data as a nested array. The transfer of data is done\n   * synchronously.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  arraySync(): ArrayMap[R] {\n    return toNestedArray(\n               this.shape, this.dataSync(), this.dtype === 'complex64') as\n        ArrayMap[R];\n  }\n\n  /**\n   * Asynchronously downloads the values from the `tf.Tensor`. Returns a\n   * promise of `TypedArray` that resolves when the computation has finished.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  async data<D extends DataType = NumericDataType>(): Promise<DataTypeMap[D]> {\n    this.throwIfDisposed();\n    const data = trackerFn().read(this.dataId);\n    if (this.dtype === 'string') {\n      const bytes = await data as Uint8Array[];\n      try {\n        return bytes.map(b => util.decodeString(b)) as DataTypeMap[D];\n      } catch {\n        throw new Error(\n            'Failed to decode the string bytes into utf-8. ' +\n            'To get the original bytes, call tensor.bytes().');\n      }\n    }\n    return data as Promise<DataTypeMap[D]>;\n  }\n\n  /**\n   * Copy the tensor's data to a new GPU resource. Comparing to the `dataSync()`\n   * and `data()`, this method prevents data from being downloaded to CPU.\n   *\n   * For WebGL backend, the data will be stored on a densely packed texture.\n   * This means that the texture will use the RGBA channels to store value.\n   *\n   * For WebGPU backend, the data will be stored on a buffer. There is no\n   * parameter, so can not use a user-defined size to create the buffer.\n   *\n   * @param options:\n   *     For WebGL,\n   *         - customTexShape: Optional. If set, will use the user defined\n   *     texture shape to create the texture.\n   *\n   * @returns For WebGL backend, a GPUData contains the new texture and\n   *     its information.\n   *     {\n   *        tensorRef: The tensor that is associated with this texture,\n   *        texture: WebGLTexture,\n   *        texShape: [number, number] // [height, width]\n   *     }\n   *\n   *     For WebGPU backend, a GPUData contains the new buffer.\n   *     {\n   *        tensorRef: The tensor that is associated with this buffer,\n   *        buffer: GPUBuffer,\n   *     }\n   *\n   *     Remember to dispose the GPUData after it is used by\n   *     `res.tensorRef.dispose()`.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  dataToGPU(options?: DataToGPUOptions): GPUData {\n    this.throwIfDisposed();\n    return trackerFn().readToGPU(this.dataId, options);\n  }\n\n  /**\n   * Synchronously downloads the values from the `tf.Tensor`. This blocks the\n   * UI thread until the values are ready, which can cause performance issues.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  dataSync<D extends DataType = NumericDataType>(): DataTypeMap[D] {\n    this.throwIfDisposed();\n    const data = trackerFn().readSync(this.dataId);\n    if (this.dtype === 'string') {\n      try {\n        return (data as Uint8Array[]).map(b => util.decodeString(b)) as\n            DataTypeMap[D];\n      } catch {\n        throw new Error(\n            'Failed to decode the string bytes into utf-8. ' +\n            'To get the original bytes, call tensor.bytes().');\n      }\n    }\n    return data as DataTypeMap[D];\n  }\n\n  /** Returns the underlying bytes of the tensor's data. */\n  async bytes(): Promise<Uint8Array[]|Uint8Array> {\n    this.throwIfDisposed();\n    const data = await trackerFn().read(this.dataId);\n    if (this.dtype === 'string') {\n      return data as Uint8Array[];\n    } else {\n      return new Uint8Array((data as TypedArray).buffer);\n    }\n  }\n\n  /**\n   * Disposes `tf.Tensor` from memory.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  dispose(): void {\n    if (this.isDisposed) {\n      return;\n    }\n    if (this.kerasMask) {\n      this.kerasMask.dispose();\n    }\n    trackerFn().disposeTensor(this);\n    this.isDisposedInternal = true;\n  }\n\n  protected isDisposedInternal = false;\n  get isDisposed(): boolean {\n    return this.isDisposedInternal;\n  }\n\n  throwIfDisposed() {\n    if (this.isDisposed) {\n      throw new Error(`Tensor is disposed.`);\n    }\n  }\n\n  /**\n   * Prints the `tf.Tensor`. See `tf.print` for details.\n   *\n   * @param verbose Whether to print verbose information about the tensor,\n   *    including dtype and size.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  print(verbose = false): void {\n    return opHandler.print(this, verbose);\n  }\n\n  /**\n   * Returns a copy of the tensor. See `tf.clone` for details.\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  clone<T extends Tensor>(this: T): T {\n    this.throwIfDisposed();\n    return opHandler.clone(this);\n  }\n\n  /**\n   * Returns a human-readable description of the tensor. Useful for logging.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  toString(verbose = false): string {\n    const vals = this.dataSync();\n    return tensorToString(vals, this.shape, this.dtype, verbose);\n  }\n\n  cast<T extends this>(dtype: DataType): T {\n    this.throwIfDisposed();\n    return opHandler.cast(this as T, dtype);\n  }\n  variable(trainable = true, name?: string, dtype?: DataType): Variable<R> {\n    this.throwIfDisposed();\n    return trackerFn().makeVariable(this, trainable, name, dtype) as\n        Variable<R>;\n  }\n}\n\nObject.defineProperty(Tensor, Symbol.hasInstance, {\n  value: (instance: Tensor) => {\n    // Implementation note: we should use properties of the object that will be\n    // defined before the constructor body has finished executing (methods).\n    // This is because when this code is transpiled by babel, babel will call\n    // classCallCheck before the constructor body is run.\n    // See https://github.com/tensorflow/tfjs/issues/3384 for backstory.\n    return !!instance && instance.data != null && instance.dataSync != null &&\n        instance.throwIfDisposed != null;\n  }\n});\n\nexport function getGlobalTensorClass() {\n  // Use getGlobal so that we can augment the Tensor class across package\n  // boundaries becase the node resolution alg may result in different modules\n  // being returned for this file depending on the path they are loaded from.\n  return getGlobal('Tensor', () => {\n    return Tensor;\n  });\n}\n\n// Global side effect. Cache global reference to Tensor class\ngetGlobalTensorClass();\n\nexport interface NumericTensor<R extends Rank = Rank> extends Tensor<R> {\n  dtype: NumericDataType;\n  dataSync<D extends DataType = NumericDataType>(): DataTypeMap[D];\n  data<D extends DataType = NumericDataType>(): Promise<DataTypeMap[D]>;\n  dataToGPU(options?: DataToGPUOptions): GPUData;\n}\n\nexport interface StringTensor<R extends Rank = Rank> extends Tensor<R> {\n  dtype: 'string';\n  dataSync<D extends DataType = 'string'>(): DataTypeMap[D];\n  data<D extends DataType = 'string'>(): Promise<DataTypeMap[D]>;\n}\n\n/** @doclink Tensor */\nexport type Scalar = Tensor<Rank.R0>;\n/** @doclink Tensor */\nexport type Tensor1D = Tensor<Rank.R1>;\n/** @doclink Tensor */\nexport type Tensor2D = Tensor<Rank.R2>;\n/** @doclink Tensor */\nexport type Tensor3D = Tensor<Rank.R3>;\n/** @doclink Tensor */\nexport type Tensor4D = Tensor<Rank.R4>;\n/** @doclink Tensor */\nexport type Tensor5D = Tensor<Rank.R5>;\n/** @doclink Tensor */\nexport type Tensor6D = Tensor<Rank.R6>;\n\n/**\n * A mutable `tf.Tensor`, useful for persisting state, e.g. for training.\n *\n * @doc {heading: 'Tensors', subheading: 'Classes'}\n */\nexport class Variable<R extends Rank = Rank> extends Tensor<R> {\n  name: string;\n\n  constructor(\n      initialValue: Tensor<R>, public trainable: boolean, name: string,\n      tensorId: number) {\n    super(\n        initialValue.shape, initialValue.dtype, initialValue.dataId, tensorId);\n    this.name = name;\n  }\n\n  /**\n   * Assign a new `tf.Tensor` to this variable. The new `tf.Tensor` must have\n   * the same shape and dtype as the old `tf.Tensor`.\n   *\n   * @param newValue New tensor to be assigned to this variable.\n   *\n   * @doc {heading: 'Tensors', subheading: 'Classes'}\n   */\n  assign(newValue: Tensor<R>): void {\n    if (newValue.dtype !== this.dtype) {\n      throw new Error(\n          `dtype of the new value (${newValue.dtype}) and ` +\n          `previous value (${this.dtype}) must match`);\n    }\n    if (!util.arraysEqual(newValue.shape, this.shape)) {\n      throw new Error(\n          `shape of the new value (${newValue.shape}) and ` +\n          `previous value (${this.shape}) must match`);\n    }\n    trackerFn().disposeTensor(this);\n    this.dataId = newValue.dataId;\n    trackerFn().incRef(this, null /* backend */);\n  }\n\n  override dispose(): void {\n    trackerFn().disposeVariable(this);\n    this.isDisposedInternal = true;\n  }\n}\n\nObject.defineProperty(Variable, Symbol.hasInstance, {\n  value: (instance: Variable) => {\n    return instance instanceof Tensor && instance.assign != null &&\n        instance.assign instanceof Function;\n  }\n});\n"]}
|