Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add primitives.containers.BitArray container class #11542

Merged
merged 9 commits into from
Jan 12, 2024
1 change: 1 addition & 0 deletions qiskit/primitives/containers/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
"""

from .bindings_array import BindingsArray
from .bit_array import BitArray
from .data_bin import make_data_bin
from .observables_array import ObservablesArray
from .primitive_result import PrimitiveResult
Expand Down
337 changes: 337 additions & 0 deletions qiskit/primitives/containers/bit_array.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,337 @@
# This code is part of Qiskit.
#
# (C) Copyright IBM 2024.
#
# This code is licensed under the Apache License, Version 2.0. You may
# obtain a copy of this license in the LICENSE.txt file in the root directory
# of this source tree or at http://www.apache.org/licenses/LICENSE-2.0.
#
# Any modifications or derivative works of this code must retain this
# copyright notice, and modified files need to carry a notice indicating
# that they have been altered from the originals.

"""
BitArray
"""

from __future__ import annotations

from typing import Callable, Dict, Iterable, Literal, Mapping, Tuple
from collections import defaultdict
from functools import partial
from itertools import chain, repeat

import numpy as np
from numpy.typing import NDArray
from qiskit.result import Counts

from .shape import ShapedMixin, ShapeInput, shape_tuple

# this lookup table tells you how many bits are 1 in each uint8 value
_WEIGHT_LOOKUP = np.unpackbits(np.arange(256, dtype=np.uint8).reshape(-1, 1), axis=1).sum(axis=1)


def _min_num_bytes(num_bits: int) -> int:
"""Return the minimum number of bytes needed to store ``num_bits``."""
return num_bits // 8 + (num_bits % 8 > 0)


class BitArray(ShapedMixin):
"""Stores an array of bit values.

This object contains a single, contiguous block of data that represents an array of bitstrings.
The last axis is over packed bits, the second last axis is over shots, and the preceding axes
correspond to the shape of the pub that was executed to sample these bits.
"""

def __init__(self, array: NDArray[np.uint8], num_bits: int):
"""
Args:
array: The ``uint8`` data array.
num_bits: How many bit are in each outcome.

Raises:
ValueError: If the input array has fewer than two axes, or the size of the last axis
is not the smallest number of bytes that can contain ``num_bits``.
"""
super().__init__()
self._array = np.array(array, copy=False, dtype=np.uint8)
chriseclectic marked this conversation as resolved.
Show resolved Hide resolved
self._num_bits = num_bits
# second last dimension is shots, last dimension is packed bits
self._shape = self._array.shape[:-2]

if self._array.ndim < 2:
raise ValueError("The input array must have at least two axes.")
if self._array.shape[-1] != (expected := _min_num_bytes(num_bits)):
raise ValueError(f"The input array is expected to have {expected} bytes per shot.")

def _prepare_broadcastable(self, other: "BitArray") -> Tuple[NDArray[np.uint8], ...]:
"""Validation and broadcasting of two bit arrays before element-wise binary operation."""
if self.num_bits != other.num_bits:
raise ValueError(f"'num_bits' must match in {self} and {other}.")
self_shape = self.shape + (self.num_shots,)
chriseclectic marked this conversation as resolved.
Show resolved Hide resolved
other_shape = other.shape + (other.num_shots,)
try:
shape = np.broadcast_shapes(self_shape, other_shape) + (self._array.shape[-1],)
except ValueError as ex:
raise ValueError(f"{self} and {other} are not compatible for this operation.") from ex
return np.broadcast_to(self.array, shape), np.broadcast_to(other.array, shape)

def __and__(self, other: "BitArray") -> "BitArray":
return BitArray(np.bitwise_and(*self._prepare_broadcastable(other)), self.num_bits)

def __eq__(self, other: "BitArray") -> bool:
if (n := self.num_bits) != other.num_bits:
return False
arrs = [self._array, other._array]
if n % 8 > 0:
# ignore straggling bits on the left
mask = np.array([255 >> ((-n) % 8)] + [255] * (n // 8), dtype=np.uint8)
arrs = [np.bitwise_and(arr, mask) for arr in arrs]
return np.array_equal(*arrs, equal_nan=False)

def __invert__(self) -> "BitArray":
return BitArray(np.bitwise_not(self._array), self.num_bits)

def __or__(self, other: "BitArray") -> "BitArray":
return BitArray(np.bitwise_or(*self._prepare_broadcastable(other)), self.num_bits)

def __xor__(self, other: "BitArray") -> "BitArray":
return BitArray(np.bitwise_xor(*self._prepare_broadcastable(other)), self.num_bits)

def __repr__(self):
desc = f"<shape={self.shape}, num_shots={self.num_shots}, num_bits={self.num_bits}>"
return f"BitArray({desc})"

@property
def array(self) -> NDArray[np.uint8]:
"""The raw NumPy array of data."""
return self._array

@property
def num_bits(self) -> int:
"""The number of bits in the register that this array stores data for.

For example, a ``ClassicalRegister(5, "meas")`` would result in ``num_bits=5``.
"""
return self._num_bits

@property
def num_shots(self) -> int:
"""The number of shots sampled from the register in each configuration.

More precisely, the length of the second last axis of :attr:`~.array`.
"""
return self._array.shape[-2]

@staticmethod
def _bytes_to_bitstring(data: bytes, num_bits: int, mask: int) -> str:
val = int.from_bytes(data, "big") & mask
return bin(val)[2:].zfill(num_bits)

@staticmethod
def _bytes_to_int(data: bytes, mask: int) -> int:
return int.from_bytes(data, "big") & mask

def _get_counts(
self, *, loc: int | Tuple[int, ...] | None, converter: Callable
) -> Dict[str | int, int]:
if loc is None and self.size == 1:
loc = (0,) * self.ndim

elif loc is None:
raise ValueError(
f"Your BitArray has shape {self.shape}, meaning that it actually represents "
f"{self.size} different count dictionaries. You need to use the `loc` argument of "
"this function to pick one of them."
)

counts = defaultdict(int)
for shot_row in self._array[loc]:
counts[converter(shot_row.tobytes())] += 1
return dict(counts)

def bitcount(self) -> NDArray[np.uint64]:
"""Compute the number of ones appearing in the binary representation of each shot.

Returns:
A ``numpy.uint64``-array with shape ``(*shape, num_shots)``.
"""
return _WEIGHT_LOOKUP[self._array].sum(axis=-1)

@staticmethod
def from_bool_array(
array: NDArray[np.bool_], order: Literal["big", "little"] = "big"
) -> "BitArray":
"""Construct a new bit array from an array of bools.

Args:
array: The array to convert, with "bitstrings" along the last axis.
order: One of ``"big"`` or ``"little"``, indicating whether ``array[..., 0]``
correspond to the most significant bits or the least significant bits of each
bitstring, respectively.

Returns:
A new bit array.
"""
array = np.array(array, dtype=bool, copy=False)
ihincks marked this conversation as resolved.
Show resolved Hide resolved

if array.ndim < 2:
raise ValueError("Expecting at least two dimensions.")

if order == "little":
# np.unpackbits assumes "big"
array = array[..., ::-1]

num_bits = array.shape[-1]
if remainder := (-num_bits) % 8:
# unpackbits pads with zeros on the wrong side with respect to what we want, so
# we manually pad to the nearest byte
pad = np.zeros(shape_tuple(array.shape[:-1], remainder), dtype=bool)
array = np.concatenate([pad, array], axis=-1)

return BitArray(np.packbits(array, axis=-1), num_bits=num_bits)

@staticmethod
def from_counts(
counts: Mapping[str | int, int] | Iterable[Mapping[str | int, int]],
num_bits: int | None = None,
) -> "BitArray":
"""Construct a new bit array from one or more ``Counts``-like objects.

The ``counts`` can have keys that are (uniformly) integers, hexstrings, or bitstrings.
Their values represent numbers of occurrences of that value.

Args:
counts: One or more counts-like mappings.
ihincks marked this conversation as resolved.
Show resolved Hide resolved
num_bits: The desired number of bits per shot. If unset, the biggest value found sets
this value.

Returns:
A new bit array.
ihincks marked this conversation as resolved.
Show resolved Hide resolved

Raises:
ValueError: If different mappings have different numbers of shots.
ValueError: If no counts dictionaries are supplied.
"""
if singleton := isinstance(counts, Mapping):
counts = [counts]
else:
counts = list(counts)
if not counts:
raise ValueError("At least one counts mapping expected.")

counts = [
mapping.int_outcomes() if isinstance(mapping, Counts) else mapping for mapping in counts
]

data = (v for mapping in counts for vs, count in mapping.items() for v in repeat(vs, count))

bit_array = BitArray.from_samples(data, num_bits)
if not singleton:
if bit_array.num_shots % len(counts) > 0:
raise ValueError("All of your mappings need to have the same number of shots.")
bit_array = bit_array.reshape(len(counts), bit_array.num_shots // len(counts))
return bit_array

@staticmethod
def from_samples(
samples: Iterable[str] | Iterable[int], num_bits: int | None = None
) -> "BitArray":
"""Construct a new bit array from an iterable of bitstrings, hexstrings, or integers.

All samples are assumed to be integers if the first one is. Strings are all assumed to be
bitstrings whenever the first string doesn't start with ``"0x"``.

Consider pairing this method with :meth:`~reshape` if your samples represent nested data.

Args:
samples: A list of bitstrings, a list of integers, or a list of hexstrings.
chriseclectic marked this conversation as resolved.
Show resolved Hide resolved
num_bits: The desired number of bits per sample. If unset, the biggest sample provided
is used to determine this value.

Returns:
A new bit array.

Raises:
ValueError: If no strings are given.
"""
samples = iter(samples)
try:
first_sample = next(samples)
except StopIteration as ex:
raise ValueError("At least one sample is required.") from ex

ints = chain([first_sample], samples)
if isinstance(first_sample, str):
base = 16 if first_sample.startswith("0x") else 2
ints = (int(val, base=base) for val in ints)

if num_bits is None:
# we are forced to prematurely look at every iterand in this case
ints = list(ints)
num_bits = max(map(int.bit_length, ints))

num_bytes = _min_num_bytes(num_bits)
data = b"".join(val.to_bytes(num_bytes, "big") for val in ints)
array = np.frombuffer(data, dtype=np.uint8, count=len(data))
return BitArray(array.reshape(-1, num_bytes), num_bits)

def get_counts(self, loc: int | Tuple[int, ...] | None = None) -> Dict[str, int]:
"""Return a counts dictionary.
ihincks marked this conversation as resolved.
Show resolved Hide resolved

Args:
loc: Which entry of this array to return a dictionary for.
chriseclectic marked this conversation as resolved.
Show resolved Hide resolved

Returns:
A dictionary mapping bitstrings to the number of occurrences of that bitstring.

Raises:
ValueError: If this array has a non-trivial size and no ``loc`` is provided.
"""
mask = 2**self.num_bits - 1
converter = partial(self._bytes_to_bitstring, num_bits=self.num_bits, mask=mask)
return self._get_counts(loc=loc, converter=converter)

def get_int_counts(self, loc: int | Tuple[int, ...] | None = None) -> Dict[int, int]:
r"""Return a counts dictionary, where bitstrings are stored as ``int``\s.

Args:
loc: Which entry of this array to return a dictionary for.

Returns:
A dictionary mapping ``ints`` to the number of occurrences of that ``int``.

Raises:
ValueError: If this array has a non-trivial size and no ``loc`` is provided.
"""
converter = partial(self._bytes_to_int, mask=2**self.num_bits - 1)
return self._get_counts(loc=loc, converter=converter)

def reshape(self, *shape: ShapeInput) -> "BitArray":
"""Return a new reshaped bit array.

The :attr:`~num_shots` axis is either included or excluded from the reshaping procedure
depending on which picture the new shape is compatible with. For example, for a bit array
with shape ``(20, 5)`` and ``64`` shots, a reshape to ``(100,)`` would leave the
number of shots intact, whereas a reshape to ``(200, 32)`` would change the number of
shots to ``32``.

Args:
*shape: The new desired shape.

Returns:
A new bit array.

Raises:
ValueError: If the size corresponding to your new shape is not equal to either
:attr:`~size`, or the product of :attr:`~size` and :attr:`~num_shots`.
"""
shape = shape_tuple(shape)
if (size := np.prod(shape, dtype=int)) == self.size:
shape = shape_tuple(shape, self._array.shape[-2:])
elif size == self.size * self.num_shots:
shape = shape_tuple(shape, self._array.shape[-1:])
else:
raise ValueError("Cannot change the size of the array.")
return BitArray(self._array.reshape(shape), self.num_bits)
Loading