This documentation is automatically generated by online-judge-tools/verification-helper
#include "Math/partizangame.hpp"
#pragma once
#include "Math/fraction.hpp"
template <typename T> struct Surreal {
static constexpr int LG = std::numeric_limits<T>::digits - 2;
Frac<T> a;
Surreal(T a = 0) : a(a, 1) {}
Surreal(T a, T b) : a(a, b) {}
Surreal(Frac<T> a) : a(a) {}
static constexpr Surreal infty() {
return Surreal(INF, 1);
}
bool operator==(Surreal const &rhs) const {
return (a == rhs.a);
}
bool operator!=(Surreal const &rhs) const {
return !(a == rhs);
}
bool operator<(Surreal const &rhs) const {
return (a < rhs.a);
}
bool operator<=(Surreal const &rhs) const {
return (a <= rhs.a);
}
bool operator>(Surreal const &rhs) const {
return (a > rhs.a);
}
bool operator>=(Surreal const &rhs) const {
return (a >= rhs.a);
}
friend Surreal operator+(const Surreal &x, const Surreal &y) {
return x.a + y.a;
}
friend Surreal operator-(const Surreal &x, const Surreal &y) {
return x.a - y.a;
}
friend Surreal operator-(const Surreal &x) {
return -x.a;
}
Surreal &operator+=(const Surreal &x) {
return (*this) = (*this) + x;
}
Surreal &operator-=(const Surreal &x) {
return (*this) = (*this) - x;
}
static Surreal between(Surreal L, Surreal R, bool incL = 0, bool incR = 0) {
Surreal ret(0);
if (L < ret or (incL and L == ret)) {
if (ret < R or (incR and R == ret)) {
return ret;
}
}
bool sign = 0;
if (R <= 0) {
sign = 1;
swap(L, R);
swap(incL, incR);
L = -L, R = -R;
}
rep(lg, 0, LG + 1) {
ll num = ceil(L.a.a << lg, L.a.b);
if ((i128(L.a.a) << lg) == i128(L.a.b) * num and !incL)
num++;
ret = Surreal(num, 1LL << lg);
if (ret < R or (incR and R == ret)) {
if (sign)
ret = -ret;
return ret;
}
}
assert(0);
}
friend ostream &operator<<(ostream &os, const Surreal &x) {
return os << x.a;
}
};
struct NumStar {
using A = Surreal<ll>;
A a;
int b;
NumStar(A a = 0, int b = 0) : a(a), b(b) {}
NumStar &operator+=(const NumStar &rhs) {
a += rhs.a, b ^= rhs.b;
return *this;
}
NumStar &operator-=(const NumStar &rhs) {
a -= rhs.a, b ^= rhs.b;
return *this;
}
NumStar operator-() const {
return NumStar(-a, b);
}
bool operator==(const NumStar &rhs) const {
return (a == rhs.a && b == rhs.b);
}
static int mex(vector<int> &a) {
vector<int> exi(SZ(a) + 1);
for (auto &x : a)
exi[x] = 1;
int ret = 0;
while (exi[ret])
ret++;
return ret;
}
static pair<bool, NumStar> calc(vector<NumStar> lb, vector<NumStar> rb) {
A L = -A::infty(), R = A::infty();
vector<int> ls, rs;
for (auto &num : lb) {
if (chmax(L, num.a))
ls.clear();
if (L == num.a)
ls.push_back(num.b);
}
for (auto &num : rb) {
if (chmin(R, num.a))
rs.clear();
if (R == num.a)
rs.push_back(num.b);
}
int lm = mex(ls), rm = mex(rs);
if (L < R) {
return {true, NumStar(A::between(L, R, lm == 0, rm == 0), 0)};
} else if (L == R) {
if (lm == rm)
return {true, NumStar(L, lm)};
}
return {false, NumStar()};
}
friend ostream &operator<<(ostream &os, const NumStar &x) {
return os << x.a << "+*" << x.b;
}
pair<bool, bool> outcome() {
if (a > 0)
return {1, 0};
if (a < 0)
return {0, 1};
if (b == 0)
return {0, 0};
return {1, 1};
}
};
// F(G):= {G_L,G_R} (pair)
template <typename State, typename F>
map<State, NumStar> SolvePartizanGame(const vector<State> &states, F option) {
map<State, NumStar> ret;
auto dfs = [&](auto &dfs, const State ¤t) -> pair<bool, NumStar> {
if (ret.count(current))
return {true, ret[current]};
auto [gl, gr] = option(current);
vector<NumStar> ls, rs;
for (auto &to : gl) {
auto [ch, lv] = dfs(dfs, to);
if (!ch)
return {false, NumStar()};
ls.push_back(lv);
}
for (auto &to : gr) {
auto [ch, rv] = dfs(dfs, to);
if (!ch)
return {false, NumStar()};
rs.push_back(rv);
}
auto [ch, val] = NumStar::calc(ls, rs);
if (!ch)
return {false, NumStar()};
return {true, ret[current] = val};
};
for (auto &s : states) {
if (!dfs(dfs, s).first)
return map<State, NumStar>();
}
return ret;
}
/**
* @brief Partizan Game Solver
*/
#line 2 "Math/partizangame.hpp"
#line 2 "Math/fraction.hpp"
template <typename T> struct Frac {
T a, b;
Frac(T _a = 0) {
init(_a, 1);
}
Frac(T _a, T _b) {
init(_a, _b);
}
Frac &init(T _a, T _b) {
T g = gcd(_a, _b);
a = _a / g, b = _b / g;
if (b < 0)
a = -a, b = -b;
return *this;
}
Frac inv() const {
return Frac(b, a);
}
Frac operator-() const {
return Frac(-a, b);
}
Frac &operator+=(const Frac &x) {
return init(a * x.b + x.a * b, b * x.b);
}
Frac &operator-=(const Frac &x) {
return init(a * x.b - x.a * b, b * x.b);
}
Frac &operator*=(const Frac &x) {
return init(a * x.a, b * x.b);
}
Frac &operator/=(const Frac &x) {
return init(a * x.b, b * x.a);
}
Frac operator+(const Frac &x) const {
return Frac(*this) += x;
}
Frac operator-(const Frac &x) const {
return Frac(*this) -= x;
}
Frac operator*(const Frac &x) const {
return Frac(*this) *= x;
}
Frac operator/(const Frac &x) const {
return Frac(*this) /= x;
}
bool operator<(const Frac &x) const {
return a * x.b < b * x.a;
}
bool operator>(const Frac &x) const {
return x < *this;
}
bool operator<=(const Frac &x) const {
return !(x < *this);
}
bool operator>=(const Frac &x) const {
return !(*this < x);
}
bool operator==(const Frac &x) const {
return (*this <= x and x <= *this);
}
bool operator!=(const Frac &x) const {
return !(*this == x);
}
friend istream &operator>>(istream &is, Frac &x) {
return is >> x.a >> x.b;
}
friend ostream &operator<<(ostream &os, const Frac &x) {
return os << x.a << '/' << x.b;
}
};
template <typename T> Frac<T> between(const Frac<T> &x, const Frac<T> &y) {
if (x.a < x.b and y.b < y.a)
return Frac(1);
else if (x.b <= x.a) {
T add = floor(x.a / x.b);
return between(x - add, y - add) + add;
} else
return between(y.inv(), x.inv()).inv();
}
/**
* @brief Fraction
* @docs docs/fraction.md
*/
#line 4 "Math/partizangame.hpp"
template <typename T> struct Surreal {
static constexpr int LG = std::numeric_limits<T>::digits - 2;
Frac<T> a;
Surreal(T a = 0) : a(a, 1) {}
Surreal(T a, T b) : a(a, b) {}
Surreal(Frac<T> a) : a(a) {}
static constexpr Surreal infty() {
return Surreal(INF, 1);
}
bool operator==(Surreal const &rhs) const {
return (a == rhs.a);
}
bool operator!=(Surreal const &rhs) const {
return !(a == rhs);
}
bool operator<(Surreal const &rhs) const {
return (a < rhs.a);
}
bool operator<=(Surreal const &rhs) const {
return (a <= rhs.a);
}
bool operator>(Surreal const &rhs) const {
return (a > rhs.a);
}
bool operator>=(Surreal const &rhs) const {
return (a >= rhs.a);
}
friend Surreal operator+(const Surreal &x, const Surreal &y) {
return x.a + y.a;
}
friend Surreal operator-(const Surreal &x, const Surreal &y) {
return x.a - y.a;
}
friend Surreal operator-(const Surreal &x) {
return -x.a;
}
Surreal &operator+=(const Surreal &x) {
return (*this) = (*this) + x;
}
Surreal &operator-=(const Surreal &x) {
return (*this) = (*this) - x;
}
static Surreal between(Surreal L, Surreal R, bool incL = 0, bool incR = 0) {
Surreal ret(0);
if (L < ret or (incL and L == ret)) {
if (ret < R or (incR and R == ret)) {
return ret;
}
}
bool sign = 0;
if (R <= 0) {
sign = 1;
swap(L, R);
swap(incL, incR);
L = -L, R = -R;
}
rep(lg, 0, LG + 1) {
ll num = ceil(L.a.a << lg, L.a.b);
if ((i128(L.a.a) << lg) == i128(L.a.b) * num and !incL)
num++;
ret = Surreal(num, 1LL << lg);
if (ret < R or (incR and R == ret)) {
if (sign)
ret = -ret;
return ret;
}
}
assert(0);
}
friend ostream &operator<<(ostream &os, const Surreal &x) {
return os << x.a;
}
};
struct NumStar {
using A = Surreal<ll>;
A a;
int b;
NumStar(A a = 0, int b = 0) : a(a), b(b) {}
NumStar &operator+=(const NumStar &rhs) {
a += rhs.a, b ^= rhs.b;
return *this;
}
NumStar &operator-=(const NumStar &rhs) {
a -= rhs.a, b ^= rhs.b;
return *this;
}
NumStar operator-() const {
return NumStar(-a, b);
}
bool operator==(const NumStar &rhs) const {
return (a == rhs.a && b == rhs.b);
}
static int mex(vector<int> &a) {
vector<int> exi(SZ(a) + 1);
for (auto &x : a)
exi[x] = 1;
int ret = 0;
while (exi[ret])
ret++;
return ret;
}
static pair<bool, NumStar> calc(vector<NumStar> lb, vector<NumStar> rb) {
A L = -A::infty(), R = A::infty();
vector<int> ls, rs;
for (auto &num : lb) {
if (chmax(L, num.a))
ls.clear();
if (L == num.a)
ls.push_back(num.b);
}
for (auto &num : rb) {
if (chmin(R, num.a))
rs.clear();
if (R == num.a)
rs.push_back(num.b);
}
int lm = mex(ls), rm = mex(rs);
if (L < R) {
return {true, NumStar(A::between(L, R, lm == 0, rm == 0), 0)};
} else if (L == R) {
if (lm == rm)
return {true, NumStar(L, lm)};
}
return {false, NumStar()};
}
friend ostream &operator<<(ostream &os, const NumStar &x) {
return os << x.a << "+*" << x.b;
}
pair<bool, bool> outcome() {
if (a > 0)
return {1, 0};
if (a < 0)
return {0, 1};
if (b == 0)
return {0, 0};
return {1, 1};
}
};
// F(G):= {G_L,G_R} (pair)
template <typename State, typename F>
map<State, NumStar> SolvePartizanGame(const vector<State> &states, F option) {
map<State, NumStar> ret;
auto dfs = [&](auto &dfs, const State ¤t) -> pair<bool, NumStar> {
if (ret.count(current))
return {true, ret[current]};
auto [gl, gr] = option(current);
vector<NumStar> ls, rs;
for (auto &to : gl) {
auto [ch, lv] = dfs(dfs, to);
if (!ch)
return {false, NumStar()};
ls.push_back(lv);
}
for (auto &to : gr) {
auto [ch, rv] = dfs(dfs, to);
if (!ch)
return {false, NumStar()};
rs.push_back(rv);
}
auto [ch, val] = NumStar::calc(ls, rs);
if (!ch)
return {false, NumStar()};
return {true, ret[current] = val};
};
for (auto &s : states) {
if (!dfs(dfs, s).first)
return map<State, NumStar>();
}
return ret;
}
/**
* @brief Partizan Game Solver
*/