Back to Pytorch

torch.Storage

docs/source/storage.rst

2.11.08.5 KB
Original Source

torch.Storage

In PyTorch, a regular tensor is a multi-dimensional array that is defined by the following components:

  • Storage: The actual data of the tensor, stored as a contiguous, one-dimensional array of bytes.
  • dtype: The data type of the elements in the tensor, such as torch.float32 or torch.int64.
  • shape: A tuple indicating the size of the tensor in each dimension.
  • Stride: The step size needed to move from one element to the next in each dimension.
  • Offset: The starting point in the storage from which the tensor data begins. This will usually be 0 for newly created tensors.

These components together define the structure and data of a tensor, with the storage holding the actual data and the rest serving as metadata.

Untyped Storage API

A :class:torch.UntypedStorage is a contiguous, one-dimensional array of elements. Its length is equal to the number of bytes of the tensor. The storage serves as the underlying data container for tensors. In general, a tensor created in PyTorch using regular constructors such as :func:~torch.zeros, :func:~torch.zeros_like or :func:~torch.Tensor.new_zeros will produce tensors where there is a one-to-one correspondence between the tensor storage and the tensor itself.

However, a storage is allowed to be shared by multiple tensors. For instance, any view of a tensor (obtained through :meth:~torch.Tensor.view or some, but not all, kinds of indexing like integers and slices) will point to the same underlying storage as the original tensor. When serializing and deserializing tensors that share a common storage, the relationship is preserved, and the tensors continue to point to the same storage. Interestingly, deserializing multiple tensors that point to a single storage can be faster than deserializing multiple independent tensors.

A tensor storage can be accessed through the :meth:~torch.Tensor.untyped_storage method. This will return an object of type :class:torch.UntypedStorage. Fortunately, storages have a unique identifier accessed through the :meth:torch.UntypedStorage.data_ptr method. In regular settings, two tensors with the same data storage will have the same storage data_ptr. However, tensors themselves can point to two separate storages, one for its data attribute and another for its grad attribute. Each will require a data_ptr() of its own. In general, there is no guarantee that a :meth:torch.Tensor.data_ptr and :meth:torch.UntypedStorage.data_ptr match and this should not be assumed to be true.

Untyped storages are somewhat independent of the tensors that are built on them. Practically, this means that tensors with different dtypes or shape can point to the same storage. It also implies that a tensor storage can be changed, as the following example shows:

>>> t = torch.ones(3)
>>> s0 = t.untyped_storage()
>>> s0
 0
 0
 128
 63
 0
 0
 128
 63
 0
 0
 128
 63
[torch.storage.UntypedStorage(device=cpu) of size 12]
>>> s1 = s0.clone()
>>> s1.fill_(0)
 0
 0
 0
 0
 0
 0
 0
 0
 0
 0
 0
 0
[torch.storage.UntypedStorage(device=cpu) of size 12]
>>> # Fill the tensor with a zeroed storage
>>> t.set_(s1, storage_offset=t.storage_offset(), stride=t.stride(), size=t.size())
tensor([0., 0., 0.])

.. warning:: Please note that directly modifying a tensor's storage as shown in this example is not a recommended practice. This low-level manipulation is illustrated solely for educational purposes, to demonstrate the relationship between tensors and their underlying storages. In general, it's more efficient and safer to use standard torch.Tensor methods, such as :meth:~torch.Tensor.clone and :meth:~torch.Tensor.fill_, to achieve the same results.

Other than data_ptr, untyped storage also have other attributes such as :attr:~torch.UntypedStorage.filename (in case the storage points to a file on disk), :attr:~torch.UntypedStorage.device or :attr:~torch.UntypedStorage.is_cuda for device checks. A storage can also be manipulated in-place or out-of-place with methods like :attr:~torch.UntypedStorage.copy_, :attr:~torch.UntypedStorage.fill_ or :attr:~torch.UntypedStorage.pin_memory. For more information, check the API reference below. Keep in mind that modifying storages is a low-level API and comes with risks! Most of these APIs also exist on the tensor level: if present, they should be prioritized over their storage counterparts.

Special cases

We mentioned that a tensor that has a non-None grad attribute has actually two pieces of data within it. In this case, :meth:~torch.Tensor.untyped_storage will return the storage of the :attr:~torch.Tensor.data attribute, whereas the storage of the gradient can be obtained through tensor.grad.untyped_storage().

>>> t = torch.zeros(3, requires_grad=True)
>>> t.sum().backward()
>>> assert list(t.untyped_storage()) == [0] * 12  # the storage of the tensor is just 0s
>>> assert list(t.grad.untyped_storage()) != [0] * 12  # the storage of the gradient isn't

There are also special cases where tensors do not have a typical storage, or no storage at all:

  • Tensors on "meta" device: Tensors on the "meta" device are used for shape inference and do not hold actual data.
  • Fake Tensors: Another internal tool used by PyTorch's compiler is FakeTensor <https://pytorch.org/docs/stable/torch.compiler_fake_tensor.html>_ which is based on a similar idea.

Tensor subclasses or tensor-like objects can also display unusual behaviours. In general, we do not expect many use cases to require operating at the Storage level!

.. autoclass:: torch.UntypedStorage :members: :undoc-members: :inherited-members:

Legacy Typed Storage

.. warning:: For historical context, PyTorch previously used typed storage classes, which are now deprecated and should be avoided. The following details this API in case you should encounter it, although its usage is highly discouraged. All storage classes except for :class:torch.UntypedStorage will be removed in the future, and :class:torch.UntypedStorage will be used in all cases.

:class:torch.Storage is an alias for the storage class that corresponds with the default data type (:func:torch.get_default_dtype()). For example, if the default data type is :attr:torch.float, :class:torch.Storage resolves to :class:torch.FloatStorage.

The :class:torch.<type>Storage and :class:torch.cuda.<type>Storage classes, like :class:torch.FloatStorage, :class:torch.IntStorage, etc., are not actually ever instantiated. Calling their constructors creates a :class:torch.TypedStorage with the appropriate :class:torch.dtype and :class:torch.device. :class:torch.<type>Storage classes have all of the same class methods that :class:torch.TypedStorage has.

A :class:torch.TypedStorage is a contiguous, one-dimensional array of elements of a particular :class:torch.dtype. It can be given any :class:torch.dtype, and the internal data will be interpreted appropriately. :class:torch.TypedStorage contains a :class:torch.UntypedStorage which holds the data as an untyped array of bytes.

Every strided :class:torch.Tensor contains a :class:torch.TypedStorage, which stores all of the data that the :class:torch.Tensor views.

.. autoclass:: torch.TypedStorage :members: :undoc-members: :inherited-members:

.. autoclass:: torch.DoubleStorage :members: :undoc-members:

.. autoclass:: torch.FloatStorage :members: :undoc-members:

.. autoclass:: torch.HalfStorage :members: :undoc-members:

.. autoclass:: torch.LongStorage :members: :undoc-members:

.. autoclass:: torch.IntStorage :members: :undoc-members:

.. autoclass:: torch.ShortStorage :members: :undoc-members:

.. autoclass:: torch.CharStorage :members: :undoc-members:

.. autoclass:: torch.ByteStorage :members: :undoc-members:

.. autoclass:: torch.BoolStorage :members: :undoc-members:

.. autoclass:: torch.BFloat16Storage :members: :undoc-members:

.. autoclass:: torch.ComplexDoubleStorage :members: :undoc-members:

.. autoclass:: torch.ComplexFloatStorage :members: :undoc-members:

.. autoclass:: torch.QUInt8Storage :members: :undoc-members:

.. autoclass:: torch.QInt8Storage :members: :undoc-members:

.. autoclass:: torch.QInt32Storage :members: :undoc-members:

.. autoclass:: torch.QUInt4x2Storage :members: :undoc-members:

.. autoclass:: torch.QUInt2x4Storage :members: :undoc-members: