This documentation is automatically generated by online-judge-tools/verification-helper
#define PROBLEM "https://judge.yosupo.jp/problem/convolution_mod"
#include "test/template.hpp"
#include "src/modint/modint.hpp"
#include "src/FPS/FFT.hpp"
int main() {
cin.tie(0)->sync_with_stdio(0);
ll N, M;
cin >> N >> M;
vector<mm> A(N), B(M);
for(mm& a : A) cin >> a.x;
for(mm& b : B) cin >> b.x;
auto C = conv(move(A), move(B));
rep(i, 0, sz(C)) cout << C[i].x << " \n"[i + 1 == sz(C)];
}
#line 1 "test/FPS/FFT.test.cpp"
#define PROBLEM "https://judge.yosupo.jp/problem/convolution_mod"
#line 1 "test/template.hpp"
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
const ll INF = LLONG_MAX / 4;
#define rep(i, a, b) for(ll i = a; i < (b); i++)
#define all(a) begin(a), end(a)
#define sz(a) ssize(a)
bool chmin(auto& a, auto b) { return a > b ? a = b, 1 : 0; }
bool chmax(auto& a, auto b) { return a < b ? a = b, 1 : 0; }
#line 1 "src/modint/modint.hpp"
const ll mod = 998244353;
struct mm {
ll x;
mm(ll x_ = 0) : x(x_ % mod) {
if(x < 0) x += mod;
}
friend mm operator+(mm a, mm b) { return a.x + b.x; }
friend mm operator-(mm a, mm b) { return a.x - b.x; }
friend mm operator*(mm a, mm b) { return a.x * b.x; }
friend mm operator/(mm a, mm b) { return a * b.inv(); }
// 4 行コピペ Alt + Shift + クリックで複数カーソル
friend mm& operator+=(mm& a, mm b) { return a = a.x + b.x; }
friend mm& operator-=(mm& a, mm b) { return a = a.x - b.x; }
friend mm& operator*=(mm& a, mm b) { return a = a.x * b.x; }
friend mm& operator/=(mm& a, mm b) { return a = a * b.inv(); }
mm inv() const { return pow(mod - 2); }
mm pow(ll b) const {
mm a = *this, c = 1;
while(b) {
if(b & 1) c *= a;
a *= a;
b >>= 1;
}
return c;
}
};
#line 1 "src/FPS/FFT.hpp"
// {998244353, 3}, {1811939329, 13}, {2013265921, 31}
mm g = 3; // 原始根
void fft(vector<mm>& a) {
ll n = sz(a), lg = __lg(n);
assert((1 << lg) == n);
vector<mm> b(n);
rep(l, 1, lg + 1) {
ll w = n >> l;
mm s = 1, r = g.pow(mod >> l);
for(ll u = 0; u < n / 2; u += w) {
rep(d, 0, w) {
mm x = a[u << 1 | d], y = a[u << 1 | w | d] * s;
b[u | d] = x + y;
b[n >> 1 | u | d] = x - y;
}
s *= r;
}
swap(a, b);
}
}
vector<mm> conv(vector<mm> a, vector<mm> b) {
if(a.empty() || b.empty()) return {};
size_t s = sz(a) + sz(b) - 1, n = bit_ceil(s);
// if(min(sz(a), sz(b)) <= 60) 愚直に掛け算
a.resize(n);
b.resize(n);
fft(a);
fft(b);
mm inv = mm(n).inv();
rep(i, 0, n) a[i] *= b[i] * inv;
reverse(1 + all(a));
fft(a);
a.resize(s);
return a;
}
#line 5 "test/FPS/FFT.test.cpp"
int main() {
cin.tie(0)->sync_with_stdio(0);
ll N, M;
cin >> N >> M;
vector<mm> A(N), B(M);
for(mm& a : A) cin >> a.x;
for(mm& b : B) cin >> b.x;
auto C = conv(move(A), move(B));
rep(i, 0, sz(C)) cout << C[i].x << " \n"[i + 1 == sz(C)];
}