mlx/mlx/random.h
Awni Hannun 40c62c1321
Use int64 stride everywhere (#1671)
* use int64 stride everywhere

* fix ext

* fix ext

* more shape + cleanup

* one more

* few more
2024-12-09 11:09:02 -08:00

271 lines
6.8 KiB
C++

// Copyright © 2023 Apple Inc.
#pragma once
#include <chrono>
#include <optional>
#include "mlx/array.h"
#include "mlx/stream.h"
#include "mlx/utils.h"
namespace mlx::core::random {
class KeySequence {
public:
explicit KeySequence(uint64_t seed);
void seed(uint64_t seed);
array next();
// static default
static KeySequence& default_() {
static KeySequence ks(get_current_time_seed());
return ks;
}
private:
array key_;
static uint64_t get_current_time_seed() {
auto now = std::chrono::system_clock::now();
return std::chrono::duration_cast<std::chrono::milliseconds>(
now.time_since_epoch())
.count();
}
};
/** Get a PRNG key from a seed. */
array key(uint64_t seed);
/** Seed the default PRNG key. */
void seed(uint64_t seed);
/** Generate an array with type uint32 filled with random bits. */
array bits(
const Shape& shape,
int width,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
inline array bits(
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return bits(shape, 4, key, s);
}
/** Split the rng key into a pair of keys. */
std::pair<array, array> split(const array& key, StreamOrDevice s = {});
/** Split the rng key into `num` keys. */
array split(const array& key, int num, StreamOrDevice s = {});
/** Generate uniform random numbers between low and high. */
array uniform(
const array& low,
const array& high,
const Shape& shape,
Dtype dtype = float32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
template <typename T, typename U>
array uniform(
T low,
U high,
const Shape& shape,
Dtype dtype = float32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return uniform(array(low), array(high), shape, dtype, key, to_stream(s));
}
/** Generate uniform random numbers between 0 and 1. */
array uniform(
const Shape& shape,
Dtype dtype,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
inline array uniform(
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return uniform(shape, float32, key);
}
/** Generate samples from the standard normal distribution. */
array normal(
const Shape& shape,
Dtype dtype,
const float loc,
const float scale,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
inline array normal(
const Shape& shape,
const float loc,
const float scale,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return normal(shape, float32, loc, scale, key, s);
}
inline array normal(
const Shape& shape,
const Dtype dtype,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return normal(shape, dtype, 0.0, 1.0, key, s);
}
inline array normal(
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return normal(shape, float32, 0.0, 1.0, key, s);
}
/** Generate samples from a multivariate normal distribution. **/
array multivariate_normal(
const array& mean,
const array& cov,
const Shape& shape,
Dtype dtype,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
/** Generate integer samples uniformly at random */
array randint(
const array& low,
const array& high,
const Shape& shape,
Dtype dtype = int32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
template <typename T, typename U>
array randint(
T low,
U high,
const Shape& shape,
Dtype dtype = int32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return randint(array(low), array(high), shape, dtype, key, to_stream(s));
}
/** Generate binary variables with probability to be true equal to p */
array bernoulli(
const array& p,
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array bernoulli(
const array& p,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
template <typename T>
array bernoulli(
T p,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return bernoulli(array(p), key, s);
}
template <typename T>
array bernoulli(
T p,
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return bernoulli(array(p), shape, key, s);
}
array bernoulli(
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array truncated_normal(
const array& lower,
const array& upper,
const Shape& shape,
Dtype dtype = float32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array truncated_normal(
const array& lower,
const array& upper,
Dtype dtype = float32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array gumbel(
const Shape& shape,
Dtype dtype = float32,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array categorical(
const array& logits,
int axis,
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array categorical(
const array& logits_,
int axis,
int num_samples,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
array categorical(
const array& logits,
int axis = -1,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
/** Generate samples from the laplace distribution. */
array laplace(
const Shape& shape,
Dtype dtype,
const float loc,
const float scale,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
inline array laplace(
const Shape& shape,
const float loc,
const float scale,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return laplace(shape, float32, loc, scale, key, s);
}
inline array laplace(
const Shape& shape,
const Dtype dtype,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return laplace(shape, dtype, 0.0, 1.0, key, s);
}
inline array laplace(
const Shape& shape,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {}) {
return laplace(shape, float32, 0.0, 1.0, key, s);
}
/* Randomly permute the elements of x along the given axis. */
array permutation(
const array& x,
int axis = 0,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
/* A random permutation of `arange(x)` */
array permutation(
int x,
const std::optional<array>& key = std::nullopt,
StreamOrDevice s = {});
} // namespace mlx::core::random