#line 1 "test/yj_suffixarray.test.cpp"
#define PROBLEM "https://judge.yosupo.jp/problem/suffixarray"
#include <cstdio>
#include <string>
#line 1 "String/sa_is.cpp"
/**
* @brief 接尾辞配列 + induced sort
* @author えびちゃん
* @see http://web.stanford.edu/class/archive/cs/cs166/cs166.1186/lectures/04/Slides04.pdf
*/
#include <cstddef>
#include <algorithm>
#include <map>
#include <utility>
#include <vector>
#line 1 "utility/literals.cpp"
/**
* @brief ユーザ定義リテラル
* @author えびちゃん
*/
#line 10 "utility/literals.cpp"
#include <cstdint>
constexpr intmax_t operator ""_jd(unsigned long long n) { return n; }
constexpr uintmax_t operator ""_ju(unsigned long long n) { return n; }
constexpr size_t operator ""_zu(unsigned long long n) { return n; }
constexpr ptrdiff_t operator ""_td(unsigned long long n) { return n; }
constexpr int8_t operator ""_i8(unsigned long long n) { return n; }
constexpr int16_t operator ""_i16(unsigned long long n) { return n; }
constexpr int32_t operator ""_i32(unsigned long long n) { return n; }
constexpr int64_t operator ""_i64(unsigned long long n) { return n; }
constexpr uint8_t operator ""_u8(unsigned long long n) { return n; }
constexpr uint16_t operator ""_u16(unsigned long long n) { return n; }
constexpr uint32_t operator ""_u32(unsigned long long n) { return n; }
constexpr uint64_t operator ""_u64(unsigned long long n) { return n; }
#line 17 "String/sa_is.cpp"
template <typename Tp>
class suffix_array {
public:
using size_type = size_t;
using difference_type = ptrdiff_t;
using value_type = Tp;
private:
std::vector<value_type> M_c;
std::vector<size_type> M_sa;
static std::vector<size_type> S_hash(std::vector<value_type> const& c) {
std::vector<size_type> h(c.size());
std::map<value_type, size_type> enc;
for (auto const& ci: c) enc[ci];
size_type m = 1;
for (auto& p: enc) p.second = m++;
for (size_type i = 0; i < c.size(); ++i) h[i] = enc.at(c[i]);
return h;
}
static std::vector<size_type> S_reduce(
std::vector<size_type> const& c, std::vector<size_type> const& lmss,
std::vector<bool> const& lms
) {
std::vector<size_type> h(lmss.size(), 0);
if (h.size() > 1) h[1] = 1;
for (size_type i = 2; i < h.size(); ++i) {
bool ident = (c[lmss[i]] == c[lmss[i-1]]);
for (size_type j = 1; ident; ++j) {
if (lms[lmss[i]+j] && lms[lmss[i-1]+j]) break;
ident &= (lms[lmss[i]+j] == lms[lmss[i-1]+j]);
if (ident) // check for boundaries
ident &= (c[lmss[i]+j] == c[lmss[i-1]+j]);
}
h[i] = (ident? h[i-1]: h[i-1]+1);
}
std::vector<size_type> rs;
std::vector<size_type> map(c.size());
for (size_type i = 0; i < lmss.size(); ++i)
map[lmss[i]] = h[i];
for (size_type i = 0; i < c.size(); ++i)
if (lms[i]) rs.push_back(map[i]);
return rs;
}
static std::vector<size_type> S_sa_is(std::vector<size_type> const& c) {
size_type n = c.size();
enum ls_type: bool { s_type, l_type };
std::vector<ls_type> ls(n);
std::vector<bool> lms(n, false);
// Label each suffix as S-type or L-type
ls[n-1] = s_type;
for (size_type i = n-1; i--;)
ls[i] = ((c[i] < c[i+1] || (c[i] == c[i+1] && ls[i+1] == s_type))? s_type: l_type);
std::vector<size_type> count(n, 0);
for (size_type i = 0; i < n; ++i) ++count[c[i]];
std::vector<size_type> sa(n, -1);
if (std::all_of(count.begin(), count.end(), [](auto x) { return x == 1; })) {
// base case
for (size_type i = 0; i < n; ++i) sa[c[i]] = i;
return sa;
}
std::vector<size_type> tail, head;
auto init_offset = [&]() {
tail = count;
for (size_type i = 1; i < n; ++i) tail[i] += tail[i-1];
head.assign(n, 0);
for (size_type i = 1; i < n; ++i) head[i] = tail[i-1];
};
auto induce = [&]() {
for (size_type i = 0; i < n; ++i) {
if (sa[i] == -1_zu || sa[i] == 0 || ls[sa[i]-1] != l_type) continue;
sa[head[c[sa[i]-1]]++] = sa[i]-1;
}
tail = count;
for (size_type i = 1; i < n; ++i) tail[i] += tail[i-1];
for (size_type i = n; i-- > 1;) {
if (sa[i] == 0 || ls[sa[i]-1] != s_type) continue;
sa[--tail[c[sa[i]-1]]] = sa[i]-1;
}
};
// Put LMS suffixes at the end of their buckets
init_offset();
for (size_type i = n; i-- > 1;) {
if (!(ls[i] == s_type && ls[i-1] == l_type)) continue;
lms[i] = true;
sa[--tail[c[i]]] = i;
}
induce();
std::vector<size_type> lmss; // in sorted order
for (size_type i = 0; i < n; ++i)
if (lms[sa[i]]) lmss.push_back(sa[i]);
// Sort LMS suffixes
auto rs_sa = S_sa_is(S_reduce(c, lmss, lms));
// Reorder LMS suffixes in the order they appear
lmss.clear();
for (size_type i = 0; i < n; ++i)
if (lms[i]) lmss.push_back(i);
// Put LMS suffixes in proper order and
init_offset();
sa.assign(n, -1_zu);
for (size_type i = rs_sa.size(); i--;) {
size_type j = lmss[rs_sa[i]];
sa[--tail[c[j]]] = j;
}
induce();
return sa;
}
void M_build_sa() {
std::vector<size_type> s = S_hash(M_c);
s.push_back(0); // for '$'
M_sa = S_sa_is(s);
}
template <typename InputIt>
bool M_lexicographical_compare(size_type pos, InputIt first, InputIt last) const {
// return true if M_c[pos:] < *[first, last)
while (first != last) {
if (pos == M_c.size()) return true;
if (M_c[pos] < *first) return true;
if (*first < M_c[pos]) return false;
++pos, ++first;
}
return false;
}
template <typename InputIt>
InputIt M_mismatch(size_type pos, InputIt first, InputIt last) const {
// with equivalence, instead of equality
while (first != last) {
if (pos == M_c.size()) break;
if (M_c[pos] < *first || *first < M_c[pos]) break;
++pos, ++first;
}
return first;
}
public:
suffix_array() = default;
suffix_array(suffix_array const&) = default;
suffix_array(suffix_array&&) = default;
template <typename InputIt>
suffix_array(InputIt first, InputIt last): M_c(first, last) { M_build_sa(); }
suffix_array& operator =(suffix_array const&) = default;
suffix_array& operator =(suffix_array&&) = default;
template <typename ForwardIt>
difference_type lcp(ForwardIt first, ForwardIt last) const {
size_type lb = 0;
size_type ub = M_c.size();
while (ub-lb > 1) {
size_type mid = (lb+ub) >> 1;
(M_lexicographical_compare(M_sa[mid], first, last)? lb: ub) = mid;
}
auto it0 = M_mismatch(M_sa[lb], first, last);
auto it1 = M_mismatch(M_sa[ub], first, last);
return std::max(std::distance(first, it0), std::distance(first, it1));
}
template <typename ForwardIt>
size_type lower_bound(ForwardIt first, ForwardIt last) const {
size_type lb = 0;
size_type ub = M_c.size();
while (ub-lb > 1) {
size_type mid = (lb+ub) >> 1;
(M_lexicographical_compare(M_sa[mid], first, last)? lb: ub) = mid;
}
if (M_lexicographical_compare(M_sa[ub], first, last)) return M_c.size();
return M_sa[ub];
}
template <typename ForwardIt>
bool contains(ForwardIt first, ForwardIt last) const {
return lcp(first, last) == std::distance(first, last);
}
size_type operator [](size_type i) const { return M_sa[i]; }
auto begin() const { return M_sa.begin(); }
auto end() const { return M_sa.end(); }
std::vector<size_type> lcp_array() const {
size_type n = M_c.size();
std::vector<size_type> rank(n+1);
for (size_type i = 0; i <= n; ++i) rank[M_sa[i]] = i;
size_type h = 0;
std::vector<size_type> lcpa(n+1, 0);
for (size_type i = 0; i < n; ++i) {
size_type j = M_sa[rank[i]-1];
if (h > 0) --h;
for (; (j+h < n && i+h < n); ++h) {
if (M_c[j+h] < M_c[i+h] || M_c[i+h] < M_c[j+h]) break;
}
lcpa[rank[i]-1] = h;
}
return lcpa;
}
};
#line 7 "test/yj_suffixarray.test.cpp"
int main() {
char buf[524288];
scanf("%s", buf);
std::string s = buf;
size_t n = s.length();
suffix_array<char> sa(s.begin(), s.end());
for (size_t i = 1; i <= n; ++i)
printf("%zu%c", sa[i], i<n? ' ': '\n');
}