#pragma once
// This file is a fork of AtCoder Library.#define E6NLAQ_CONVOLUTION_HPP
#include<algorithm>
#include<array>
#include<cassert>
#include<type_traits>
#include<vector>#include"internal/bit.hpp"
#include"modint.hpp"namespacee6nlaq{namespaceinternal{template<classmint,intg=internal::primitive_root<mint::mod()>,internal::is_static_modint_t<mint>*=nullptr>structfft_info{staticconstexprintrank2=countr_zero_constexpr(mint::mod()-1);std::array<mint,rank2+1>root;// root[i]^(2^i) == 1std::array<mint,rank2+1>iroot;// root[i] * iroot[i] == 1std::array<mint,std::max(0,rank2-2+1)>rate2;std::array<mint,std::max(0,rank2-2+1)>irate2;std::array<mint,std::max(0,rank2-3+1)>rate3;std::array<mint,std::max(0,rank2-3+1)>irate3;fft_info(){root[rank2]=mint(g).pow((mint::mod()-1)>>rank2);iroot[rank2]=root[rank2].inv();for(inti=rank2-1;i>=0;i--){root[i]=root[i+1]*root[i+1];iroot[i]=iroot[i+1]*iroot[i+1];}{mintprod=1,iprod=1;for(inti=0;i<=rank2-2;i++){rate2[i]=root[i+2]*prod;irate2[i]=iroot[i+2]*iprod;prod*=iroot[i+2];iprod*=root[i+2];}}{mintprod=1,iprod=1;for(inti=0;i<=rank2-3;i++){rate3[i]=root[i+3]*prod;irate3[i]=iroot[i+3]*iprod;prod*=iroot[i+3];iprod*=root[i+3];}}}};template<classmint,internal::is_static_modint_t<mint>*=nullptr>voidbutterfly(std::vector<mint>&a){intn=int(a.size());inth=internal::countr_zero((unsignedint)n);staticconstfft_info<mint>info;intlen=0;// a[i, i+(n>>len), i+2*(n>>len), ..] is transformedwhile(len<h){if(h-len==1){intp=1<<(h-len-1);mintrot=1;for(ints=0;s<(1<<len);s++){intoffset=s<<(h-len);for(inti=0;i<p;i++){autol=a[i+offset];autor=a[i+offset+p]*rot;a[i+offset]=l+r;a[i+offset+p]=l-r;}if(s+1!=(1<<len))rot*=info.rate2[countr_zero(~(unsignedint)(s))];}len++;}else{// 4-baseintp=1<<(h-len-2);mintrot=1,imag=info.root[2];for(ints=0;s<(1<<len);s++){mintrot2=rot*rot;mintrot3=rot2*rot;intoffset=s<<(h-len);for(inti=0;i<p;i++){automod2=1ULL*mint::mod()*mint::mod();autoa0=1ULL*a[i+offset].val();autoa1=1ULL*a[i+offset+p].val()*rot.val();autoa2=1ULL*a[i+offset+2*p].val()*rot2.val();autoa3=1ULL*a[i+offset+3*p].val()*rot3.val();autoa1na3imag=1ULL*mint(a1+mod2-a3).val()*imag.val();autona2=mod2-a2;a[i+offset]=a0+a2+a1+a3;a[i+offset+1*p]=a0+a2+(2*mod2-(a1+a3));a[i+offset+2*p]=a0+na2+a1na3imag;a[i+offset+3*p]=a0+na2+(mod2-a1na3imag);}if(s+1!=(1<<len))rot*=info.rate3[countr_zero(~(unsignedint)(s))];}len+=2;}}}template<classmint,internal::is_static_modint_t<mint>*=nullptr>voidbutterfly_inv(std::vector<mint>&a){intn=int(a.size());inth=internal::countr_zero((unsignedint)n);staticconstfft_info<mint>info;intlen=h;// a[i, i+(n>>len), i+2*(n>>len), ..] is transformedwhile(len){if(len==1){intp=1<<(h-len);mintirot=1;for(ints=0;s<(1<<(len-1));s++){intoffset=s<<(h-len+1);for(inti=0;i<p;i++){autol=a[i+offset];autor=a[i+offset+p];a[i+offset]=l+r;a[i+offset+p]=(unsignedlonglong)(mint::mod()+l.val()-r.val())*irot.val();;}if(s+1!=(1<<(len-1)))irot*=info.irate2[countr_zero(~(unsignedint)(s))];}len--;}else{// 4-baseintp=1<<(h-len);mintirot=1,iimag=info.iroot[2];for(ints=0;s<(1<<(len-2));s++){mintirot2=irot*irot;mintirot3=irot2*irot;intoffset=s<<(h-len+2);for(inti=0;i<p;i++){autoa0=1ULL*a[i+offset+0*p].val();autoa1=1ULL*a[i+offset+1*p].val();autoa2=1ULL*a[i+offset+2*p].val();autoa3=1ULL*a[i+offset+3*p].val();autoa2na3iimag=1ULL*mint((mint::mod()+a2-a3)*iimag.val()).val();a[i+offset]=a0+a1+a2+a3;a[i+offset+1*p]=(a0+(mint::mod()-a1)+a2na3iimag)*irot.val();a[i+offset+2*p]=(a0+a1+(mint::mod()-a2)+(mint::mod()-a3))*irot2.val();a[i+offset+3*p]=(a0+(mint::mod()-a1)+(mint::mod()-a2na3iimag))*irot3.val();}if(s+1!=(1<<(len-2)))irot*=info.irate3[countr_zero(~(unsignedint)(s))];}len-=2;}}}template<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution_naive(conststd::vector<mint>&a,conststd::vector<mint>&b){intn=int(a.size()),m=int(b.size());std::vector<mint>ans(n+m-1);if(n<m){for(intj=0;j<m;j++){for(inti=0;i<n;i++){ans[i+j]+=a[i]*b[j];}}}else{for(inti=0;i<n;i++){for(intj=0;j<m;j++){ans[i+j]+=a[i]*b[j];}}}returnans;}template<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution_fft(std::vector<mint>a,std::vector<mint>b){intn=int(a.size()),m=int(b.size());intz=(int)internal::bit_ceil((unsignedint)(n+m-1));a.resize(z);internal::butterfly(a);b.resize(z);internal::butterfly(b);for(inti=0;i<z;i++){a[i]*=b[i];}internal::butterfly_inv(a);a.resize(n+m-1);mintiz=mint(z).inv();for(inti=0;i<n+m-1;i++)a[i]*=iz;returna;}}// namespace internaltemplate<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution(std::vector<mint>&&a,std::vector<mint>&&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};intz=(int)internal::bit_ceil((unsignedint)(n+m-1));assert((mint::mod()-1)%z==0);if(std::min(n,m)<=60)returnconvolution_naive(a,b);returninternal::convolution_fft(a,b);}template<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution(conststd::vector<mint>&a,conststd::vector<mint>&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};intz=(int)internal::bit_ceil((unsignedint)(n+m-1));assert((mint::mod()-1)%z==0);if(std::min(n,m)<=60)returnconvolution_naive(a,b);returninternal::convolution_fft(a,b);}template<unsignedintmod=998244353,classT,std::enable_if_t<internal::is_integral<T>::value>*=nullptr>std::vector<T>convolution(conststd::vector<T>&a,conststd::vector<T>&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};usingmint=static_modint<mod>;intz=(int)internal::bit_ceil((unsignedint)(n+m-1));assert((mint::mod()-1)%z==0);std::vector<mint>a2(n),b2(m);for(inti=0;i<n;i++){a2[i]=mint(a[i]);}for(inti=0;i<m;i++){b2[i]=mint(b[i]);}autoc2=convolution(std::move(a2),std::move(b2));std::vector<T>c(n+m-1);for(inti=0;i<n+m-1;i++){c[i]=c2[i].val();}returnc;}std::vector<longlong>convolution_ll(conststd::vector<longlong>&a,conststd::vector<longlong>&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};staticconstexprunsignedlonglongMOD1=754974721;// 2^24staticconstexprunsignedlonglongMOD2=167772161;// 2^25staticconstexprunsignedlonglongMOD3=469762049;// 2^26staticconstexprunsignedlonglongM2M3=MOD2*MOD3;staticconstexprunsignedlonglongM1M3=MOD1*MOD3;staticconstexprunsignedlonglongM1M2=MOD1*MOD2;staticconstexprunsignedlonglongM1M2M3=MOD1*MOD2*MOD3;staticconstexprunsignedlonglongi1=internal::inv_gcd(MOD2*MOD3,MOD1).second;staticconstexprunsignedlonglongi2=internal::inv_gcd(MOD1*MOD3,MOD2).second;staticconstexprunsignedlonglongi3=internal::inv_gcd(MOD1*MOD2,MOD3).second;staticconstexprintMAX_AB_BIT=24;static_assert(MOD1%(1ull<<MAX_AB_BIT)==1,"MOD1 isn't enough to support an array length of 2^24.");static_assert(MOD2%(1ull<<MAX_AB_BIT)==1,"MOD2 isn't enough to support an array length of 2^24.");static_assert(MOD3%(1ull<<MAX_AB_BIT)==1,"MOD3 isn't enough to support an array length of 2^24.");assert(n+m-1<=(1<<MAX_AB_BIT));autoc1=convolution<MOD1>(a,b);autoc2=convolution<MOD2>(a,b);autoc3=convolution<MOD3>(a,b);std::vector<longlong>c(n+m-1);for(inti=0;i<n+m-1;i++){unsignedlonglongx=0;x+=(c1[i]*i1)%MOD1*M2M3;x+=(c2[i]*i2)%MOD2*M1M3;x+=(c3[i]*i3)%MOD3*M1M2;// B = 2^63, -B <= x, r(real value) < B// (x, x - M, x - 2M, or x - 3M) = r (mod 2B)// r = c1[i] (mod MOD1)// focus on MOD1// r = x, x - M', x - 2M', x - 3M' (M' = M % 2^64) (mod 2B)// r = x,// x - M' + (0 or 2B),// x - 2M' + (0, 2B or 4B),// x - 3M' + (0, 2B, 4B or 6B) (without mod!)// (r - x) = 0, (0)// - M' + (0 or 2B), (1)// -2M' + (0 or 2B or 4B), (2)// -3M' + (0 or 2B or 4B or 6B) (3) (mod MOD1)// we checked that// ((1) mod MOD1) mod 5 = 2// ((2) mod MOD1) mod 5 = 3// ((3) mod MOD1) mod 5 = 4longlongdiff=c1[i]-internal::safe_mod((longlong)(x),(longlong)(MOD1));if(diff<0)diff+=MOD1;staticconstexprunsignedlonglongoffset[5]={0,0,M1M2M3,2*M1M2M3,3*M1M2M3};x-=offset[diff%5];c[i]=x;}returnc;}}// namespace e6nlaq
#line 2 "include/e6nlaq/convolution.hpp"
// This file is a fork of AtCoder Library.#define E6NLAQ_CONVOLUTION_HPP
#include<algorithm>
#include<array>
#include<cassert>
#include<type_traits>
#include<vector>#line 2 "include/e6nlaq/internal/bit.hpp"
// This file is a fork of AtCoder Library.#define E6NLAQ_INTERNAL_BITOP_HPP
#ifdef _MSC_VER
#include<intrin.h>
#endif
#if __cplusplus >= 202002L
#include<bit>
#endif
namespacee6nlaq{namespaceinternal{#if __cplusplus >= 202002L
usingstd::bit_ceil;#else
// @return same with std::bit::bit_ceilunsignedintbit_ceil(unsignedintn){unsignedintx=1;while(x<(unsignedint)(n))x*=2;returnx;}#endif
// @param n `1 <= n`// @return same with std::bit::countr_zerointcountr_zero(unsignedintn){#ifdef _MSC_VER
unsignedlongindex;_BitScanForward(&index,n);returnindex;#else
return__builtin_ctz(n);#endif
}// @param n `1 <= n`// @return same with std::bit::countr_zeroconstexprintcountr_zero_constexpr(unsignedintn){intx=0;while(!(n&(1<<x)))x++;returnx;}}// namespace internal}// namespace e6nlaq#line 2 "include/e6nlaq/modint.hpp"
// This file is a fork of AtCoder Library.#define E6NLAQ_MODINT_HPP
#line 9 "include/e6nlaq/modint.hpp"
#line 2 "include/e6nlaq/internal/math.hpp"
// This file is a fork of AtCoder Library.#define E6NLAQ_INTERNAL_MATH_HPP
#include<utility>#ifdef _MSC_VER
#include<intrin.h>
#endif
namespacee6nlaq{namespaceinternal{// @param m `1 <= m`// @return x mod mconstexprlonglongsafe_mod(longlongx,longlongm){x%=m;if(x<0)x+=m;returnx;}// Fast modular multiplication by barrett reduction// Reference: https://en.wikipedia.org/wiki/Barrett_reduction// NOTE: reconsider after Ice Lakestructbarrett{unsignedint_m;unsignedlonglongim;// @param m `1 <= m`explicitbarrett(unsignedintm):_m(m),im((unsignedlonglong)(-1)/m+1){}// @return munsignedintumod()const{return_m;}// @param a `0 <= a < m`// @param b `0 <= b < m`// @return `a * b % m`unsignedintmul(unsignedinta,unsignedintb)const{// [1] m = 1// a = b = im = 0, so okay// [2] m >= 2// im = ceil(2^64 / m)// -> im * m = 2^64 + r (0 <= r < m)// let z = a*b = c*m + d (0 <= c, d < m)// a*b * im = (c*m + d) * im = c*(im*m) + d*im = c*2^64 + c*r + d*im// c*r + d*im < m * m + m * im < m * m + 2^64 + m <= 2^64 + m * (m + 1) < 2^64 * 2// ((ab * im) >> 64) == c or c + 1unsignedlonglongz=a;z*=b;#ifdef _MSC_VER
unsignedlonglongx;_umul128(z,im,&x);#else
unsignedlonglongx=(unsignedlonglong)(((unsigned__int128)(z)*im)>>64);#endif
unsignedlonglongy=x*_m;return(unsignedint)(z-y+(z<y?_m:0));}};// @param n `0 <= n`// @param m `1 <= m`// @return `(x ** n) % m`constexprlonglongpow_mod_constexpr(longlongx,longlongn,intm){if(m==1)return0;unsignedint_m=(unsignedint)(m);unsignedlonglongr=1;unsignedlonglongy=safe_mod(x,m);while(n){if(n&1)r=(r*y)%_m;y=(y*y)%_m;n>>=1;}returnr;}// Reference:// M. Forisek and J. Jancina,// Fast Primality Testing for Integers That Fit into a Machine Word// @param n `0 <= n`constexprboolis_prime_constexpr(intn){if(n<=1)returnfalse;if(n==2||n==7||n==61)returntrue;if(n%2==0)returnfalse;longlongd=n-1;while(d%2==0)d/=2;constexprlonglongbases[3]={2,7,61};for(longlonga:bases){longlongt=d;longlongy=pow_mod_constexpr(a,t,n);while(t!=n-1&&y!=1&&y!=n-1){y=y*y%n;t<<=1;}if(y!=n-1&&t%2==0){returnfalse;}}returntrue;}template<intn>constexprboolis_prime=is_prime_constexpr(n);// @param b `1 <= b`// @return pair(g, x) s.t. g = gcd(a, b), xa = g (mod b), 0 <= x < b/gconstexprstd::pair<longlong,longlong>inv_gcd(longlonga,longlongb){a=safe_mod(a,b);if(a==0)return{b,0};// Contracts:// [1] s - m0 * a = 0 (mod b)// [2] t - m1 * a = 0 (mod b)// [3] s * |m1| + t * |m0| <= blonglongs=b,t=a;longlongm0=0,m1=1;while(t){longlongu=s/t;s-=t*u;m0-=m1*u;// |m1 * u| <= |m1| * s <= b// [3]:// (s - t * u) * |m1| + t * |m0 - m1 * u|// <= s * |m1| - t * u * |m1| + t * (|m0| + |m1| * u)// = s * |m1| + t * |m0| <= bautotmp=s;s=t;t=tmp;tmp=m0;m0=m1;m1=tmp;}// by [3]: |m0| <= b/g// by g != b: |m0| < b/gif(m0<0)m0+=b/s;return{s,m0};}// Compile time primitive root// @param m must be prime// @return primitive root (and minimum in now)constexprintprimitive_root_constexpr(intm){if(m==2)return1;if(m==167772161)return3;if(m==469762049)return3;if(m==754974721)return11;if(m==998244353)return3;intdivs[20]={};divs[0]=2;intcnt=1;intx=(m-1)/2;while(x%2==0)x/=2;for(inti=3;(longlong)(i)*i<=x;i+=2){if(x%i==0){divs[cnt++]=i;while(x%i==0){x/=i;}}}if(x>1){divs[cnt++]=x;}for(intg=2;;g++){boolok=true;for(inti=0;i<cnt;i++){if(pow_mod_constexpr(g,(m-1)/divs[i],m)==1){ok=false;break;}}if(ok)returng;}}template<intm>constexprintprimitive_root=primitive_root_constexpr(m);// @param n `n < 2^32`// @param m `1 <= m < 2^32`// @return sum_{i=0}^{n-1} floor((ai + b) / m) (mod 2^64)unsignedlonglongfloor_sum_unsigned(unsignedlonglongn,unsignedlonglongm,unsignedlonglonga,unsignedlonglongb){unsignedlonglongans=0;while(true){if(a>=m){ans+=n*(n-1)/2*(a/m);a%=m;}if(b>=m){ans+=n*(b/m);b%=m;}unsignedlonglongy_max=a*n+b;if(y_max<m)break;// y_max < m * (n + 1)// floor(y_max / m) <= nn=(unsignedlonglong)(y_max/m);b=(unsignedlonglong)(y_max%m);std::swap(m,a);}returnans;}}// namespace internal}// namespace e6nlaq#line 2 "include/e6nlaq/internal/type_traits.hpp"
// This file is a fork of AtCoder Library.#define E6NLAQ_INTERNAL_TYPE_TRAITS_HPP
#line 9 "include/e6nlaq/internal/type_traits.hpp"
namespacee6nlaq{namespaceinternal{#ifndef _MSC_VER
template<classT>usingis_signed_int128=typenamestd::conditional<std::is_same<T,__int128_t>::value||std::is_same<T,__int128>::value,std::true_type,std::false_type>::type;template<classT>usingis_unsigned_int128=typenamestd::conditional<std::is_same<T,__uint128_t>::value||std::is_same<T,unsigned__int128>::value,std::true_type,std::false_type>::type;template<classT>usingmake_unsigned_int128=typenamestd::conditional<std::is_same<T,__int128_t>::value,__uint128_t,unsigned__int128>;template<classT>usingis_integral=typenamestd::conditional<std::is_integral<T>::value||is_signed_int128<T>::value||is_unsigned_int128<T>::value,std::true_type,std::false_type>::type;template<classT>usingis_signed_int=typenamestd::conditional<(is_integral<T>::value&&std::is_signed<T>::value)||is_signed_int128<T>::value,std::true_type,std::false_type>::type;template<classT>usingis_unsigned_int=typenamestd::conditional<(is_integral<T>::value&&std::is_unsigned<T>::value)||is_unsigned_int128<T>::value,std::true_type,std::false_type>::type;template<classT>usingto_unsigned=typenamestd::conditional<is_signed_int128<T>::value,make_unsigned_int128<T>,typenamestd::conditional<std::is_signed<T>::value,std::make_unsigned<T>,std::common_type<T>>::type>::type;#else
template<classT>usingis_integral=typenamestd::is_integral<T>;template<classT>usingis_signed_int=typenamestd::conditional<is_integral<T>::value&&std::is_signed<T>::value,std::true_type,std::false_type>::type;template<classT>usingis_unsigned_int=typenamestd::conditional<is_integral<T>::value&&std::is_unsigned<T>::value,std::true_type,std::false_type>::type;template<classT>usingto_unsigned=typenamestd::conditional<is_signed_int<T>::value,std::make_unsigned<T>,std::common_type<T>>::type;#endif
template<classT>usingis_signed_int_t=std::enable_if_t<is_signed_int<T>::value>;template<classT>usingis_unsigned_int_t=std::enable_if_t<is_unsigned_int<T>::value>;template<classT>usingto_unsigned_t=typenameto_unsigned<T>::type;}// namespace internal}// namespace e6nlaq#line 12 "include/e6nlaq/modint.hpp"
namespacee6nlaq{namespaceinternal{structmodint_base{};structstatic_modint_base:modint_base{};template<classT>usingis_modint=std::is_base_of<modint_base,T>;template<classT>usingis_modint_t=std::enable_if_t<is_modint<T>::value>;}// namespace internaltemplate<intm,std::enable_if_t<(1<=m)>*=nullptr>structstatic_modint:internal::static_modint_base{usingmint=static_modint;public:staticconstexprintmod(){returnm;}staticmintraw(intv){mintx;x._v=v;returnx;}static_modint():_v(0){}template<classT,internal::is_signed_int_t<T>*=nullptr>static_modint(Tv){longlongx=(longlong)(v%(longlong)(umod()));if(x<0)x+=umod();_v=(unsignedint)(x);}template<classT,internal::is_unsigned_int_t<T>*=nullptr>static_modint(Tv){_v=(unsignedint)(v%umod());}unsignedintval()const{return_v;}mint&operator++(){_v++;if(_v==umod())_v=0;return*this;}mint&operator--(){if(_v==0)_v=umod();_v--;return*this;}mintoperator++(int){mintresult=*this;++*this;returnresult;}mintoperator--(int){mintresult=*this;--*this;returnresult;}mint&operator+=(constmint&rhs){_v+=rhs._v;if(_v>=umod())_v-=umod();return*this;}mint&operator-=(constmint&rhs){_v-=rhs._v;if(_v>=umod())_v+=umod();return*this;}mint&operator*=(constmint&rhs){unsignedlonglongz=_v;z*=rhs._v;_v=(unsignedint)(z%umod());return*this;}mint&operator/=(constmint&rhs){return*this=*this*rhs.inv();}mintoperator+()const{return*this;}mintoperator-()const{returnmint()-*this;}mintpow(longlongn)const{assert(0<=n);mintx=*this,r=1;while(n){if(n&1)r*=x;x*=x;n>>=1;}returnr;}mintinv()const{if(prime){assert(_v);returnpow(umod()-2);}else{autoeg=internal::inv_gcd(_v,m);assert(eg.first==1);returneg.second;}}friendmintoperator+(constmint&lhs,constmint&rhs){returnmint(lhs)+=rhs;}friendmintoperator-(constmint&lhs,constmint&rhs){returnmint(lhs)-=rhs;}friendmintoperator*(constmint&lhs,constmint&rhs){returnmint(lhs)*=rhs;}friendmintoperator/(constmint&lhs,constmint&rhs){returnmint(lhs)/=rhs;}friendbooloperator==(constmint&lhs,constmint&rhs){returnlhs._v==rhs._v;}friendbooloperator!=(constmint&lhs,constmint&rhs){returnlhs._v!=rhs._v;}private:unsignedint_v;staticconstexprunsignedintumod(){returnm;}staticconstexprboolprime=internal::is_prime<m>;};template<intid>structdynamic_modint:internal::modint_base{usingmint=dynamic_modint;public:staticintmod(){return(int)(bt.umod());}staticvoidset_mod(intm){assert(1<=m);bt=internal::barrett(m);}staticmintraw(intv){mintx;x._v=v;returnx;}dynamic_modint():_v(0){}template<classT,internal::is_signed_int_t<T>*=nullptr>dynamic_modint(Tv){longlongx=(longlong)(v%(longlong)(mod()));if(x<0)x+=mod();_v=(unsignedint)(x);}template<classT,internal::is_unsigned_int_t<T>*=nullptr>dynamic_modint(Tv){_v=(unsignedint)(v%mod());}unsignedintval()const{return_v;}mint&operator++(){_v++;if(_v==umod())_v=0;return*this;}mint&operator--(){if(_v==0)_v=umod();_v--;return*this;}mintoperator++(int){mintresult=*this;++*this;returnresult;}mintoperator--(int){mintresult=*this;--*this;returnresult;}mint&operator+=(constmint&rhs){_v+=rhs._v;if(_v>=umod())_v-=umod();return*this;}mint&operator-=(constmint&rhs){_v+=mod()-rhs._v;if(_v>=umod())_v-=umod();return*this;}mint&operator*=(constmint&rhs){_v=bt.mul(_v,rhs._v);return*this;}mint&operator/=(constmint&rhs){return*this=*this*rhs.inv();}mintoperator+()const{return*this;}mintoperator-()const{returnmint()-*this;}mintpow(longlongn)const{assert(0<=n);mintx=*this,r=1;while(n){if(n&1)r*=x;x*=x;n>>=1;}returnr;}mintinv()const{autoeg=internal::inv_gcd(_v,mod());assert(eg.first==1);returneg.second;}friendmintoperator+(constmint&lhs,constmint&rhs){returnmint(lhs)+=rhs;}friendmintoperator-(constmint&lhs,constmint&rhs){returnmint(lhs)-=rhs;}friendmintoperator*(constmint&lhs,constmint&rhs){returnmint(lhs)*=rhs;}friendmintoperator/(constmint&lhs,constmint&rhs){returnmint(lhs)/=rhs;}friendbooloperator==(constmint&lhs,constmint&rhs){returnlhs._v==rhs._v;}friendbooloperator!=(constmint&lhs,constmint&rhs){returnlhs._v!=rhs._v;}private:unsignedint_v;staticinternal::barrettbt;staticunsignedintumod(){returnbt.umod();}};template<intid>internal::barrettdynamic_modint<id>::bt(998244353);usingmodint998244353=static_modint<998244353>;usingmodint1000000007=static_modint<1000000007>;usingmodint=dynamic_modint<-1>;namespaceinternal{template<classT>usingis_static_modint=std::is_base_of<internal::static_modint_base,T>;template<classT>usingis_static_modint_t=std::enable_if_t<is_static_modint<T>::value>;template<class>structis_dynamic_modint:publicstd::false_type{};template<intid>structis_dynamic_modint<dynamic_modint<id>>:publicstd::true_type{};template<classT>usingis_dynamic_modint_t=std::enable_if_t<is_dynamic_modint<T>::value>;}// namespace internal}// namespace e6nlaq#line 15 "include/e6nlaq/convolution.hpp"
namespacee6nlaq{namespaceinternal{template<classmint,intg=internal::primitive_root<mint::mod()>,internal::is_static_modint_t<mint>*=nullptr>structfft_info{staticconstexprintrank2=countr_zero_constexpr(mint::mod()-1);std::array<mint,rank2+1>root;// root[i]^(2^i) == 1std::array<mint,rank2+1>iroot;// root[i] * iroot[i] == 1std::array<mint,std::max(0,rank2-2+1)>rate2;std::array<mint,std::max(0,rank2-2+1)>irate2;std::array<mint,std::max(0,rank2-3+1)>rate3;std::array<mint,std::max(0,rank2-3+1)>irate3;fft_info(){root[rank2]=mint(g).pow((mint::mod()-1)>>rank2);iroot[rank2]=root[rank2].inv();for(inti=rank2-1;i>=0;i--){root[i]=root[i+1]*root[i+1];iroot[i]=iroot[i+1]*iroot[i+1];}{mintprod=1,iprod=1;for(inti=0;i<=rank2-2;i++){rate2[i]=root[i+2]*prod;irate2[i]=iroot[i+2]*iprod;prod*=iroot[i+2];iprod*=root[i+2];}}{mintprod=1,iprod=1;for(inti=0;i<=rank2-3;i++){rate3[i]=root[i+3]*prod;irate3[i]=iroot[i+3]*iprod;prod*=iroot[i+3];iprod*=root[i+3];}}}};template<classmint,internal::is_static_modint_t<mint>*=nullptr>voidbutterfly(std::vector<mint>&a){intn=int(a.size());inth=internal::countr_zero((unsignedint)n);staticconstfft_info<mint>info;intlen=0;// a[i, i+(n>>len), i+2*(n>>len), ..] is transformedwhile(len<h){if(h-len==1){intp=1<<(h-len-1);mintrot=1;for(ints=0;s<(1<<len);s++){intoffset=s<<(h-len);for(inti=0;i<p;i++){autol=a[i+offset];autor=a[i+offset+p]*rot;a[i+offset]=l+r;a[i+offset+p]=l-r;}if(s+1!=(1<<len))rot*=info.rate2[countr_zero(~(unsignedint)(s))];}len++;}else{// 4-baseintp=1<<(h-len-2);mintrot=1,imag=info.root[2];for(ints=0;s<(1<<len);s++){mintrot2=rot*rot;mintrot3=rot2*rot;intoffset=s<<(h-len);for(inti=0;i<p;i++){automod2=1ULL*mint::mod()*mint::mod();autoa0=1ULL*a[i+offset].val();autoa1=1ULL*a[i+offset+p].val()*rot.val();autoa2=1ULL*a[i+offset+2*p].val()*rot2.val();autoa3=1ULL*a[i+offset+3*p].val()*rot3.val();autoa1na3imag=1ULL*mint(a1+mod2-a3).val()*imag.val();autona2=mod2-a2;a[i+offset]=a0+a2+a1+a3;a[i+offset+1*p]=a0+a2+(2*mod2-(a1+a3));a[i+offset+2*p]=a0+na2+a1na3imag;a[i+offset+3*p]=a0+na2+(mod2-a1na3imag);}if(s+1!=(1<<len))rot*=info.rate3[countr_zero(~(unsignedint)(s))];}len+=2;}}}template<classmint,internal::is_static_modint_t<mint>*=nullptr>voidbutterfly_inv(std::vector<mint>&a){intn=int(a.size());inth=internal::countr_zero((unsignedint)n);staticconstfft_info<mint>info;intlen=h;// a[i, i+(n>>len), i+2*(n>>len), ..] is transformedwhile(len){if(len==1){intp=1<<(h-len);mintirot=1;for(ints=0;s<(1<<(len-1));s++){intoffset=s<<(h-len+1);for(inti=0;i<p;i++){autol=a[i+offset];autor=a[i+offset+p];a[i+offset]=l+r;a[i+offset+p]=(unsignedlonglong)(mint::mod()+l.val()-r.val())*irot.val();;}if(s+1!=(1<<(len-1)))irot*=info.irate2[countr_zero(~(unsignedint)(s))];}len--;}else{// 4-baseintp=1<<(h-len);mintirot=1,iimag=info.iroot[2];for(ints=0;s<(1<<(len-2));s++){mintirot2=irot*irot;mintirot3=irot2*irot;intoffset=s<<(h-len+2);for(inti=0;i<p;i++){autoa0=1ULL*a[i+offset+0*p].val();autoa1=1ULL*a[i+offset+1*p].val();autoa2=1ULL*a[i+offset+2*p].val();autoa3=1ULL*a[i+offset+3*p].val();autoa2na3iimag=1ULL*mint((mint::mod()+a2-a3)*iimag.val()).val();a[i+offset]=a0+a1+a2+a3;a[i+offset+1*p]=(a0+(mint::mod()-a1)+a2na3iimag)*irot.val();a[i+offset+2*p]=(a0+a1+(mint::mod()-a2)+(mint::mod()-a3))*irot2.val();a[i+offset+3*p]=(a0+(mint::mod()-a1)+(mint::mod()-a2na3iimag))*irot3.val();}if(s+1!=(1<<(len-2)))irot*=info.irate3[countr_zero(~(unsignedint)(s))];}len-=2;}}}template<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution_naive(conststd::vector<mint>&a,conststd::vector<mint>&b){intn=int(a.size()),m=int(b.size());std::vector<mint>ans(n+m-1);if(n<m){for(intj=0;j<m;j++){for(inti=0;i<n;i++){ans[i+j]+=a[i]*b[j];}}}else{for(inti=0;i<n;i++){for(intj=0;j<m;j++){ans[i+j]+=a[i]*b[j];}}}returnans;}template<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution_fft(std::vector<mint>a,std::vector<mint>b){intn=int(a.size()),m=int(b.size());intz=(int)internal::bit_ceil((unsignedint)(n+m-1));a.resize(z);internal::butterfly(a);b.resize(z);internal::butterfly(b);for(inti=0;i<z;i++){a[i]*=b[i];}internal::butterfly_inv(a);a.resize(n+m-1);mintiz=mint(z).inv();for(inti=0;i<n+m-1;i++)a[i]*=iz;returna;}}// namespace internaltemplate<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution(std::vector<mint>&&a,std::vector<mint>&&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};intz=(int)internal::bit_ceil((unsignedint)(n+m-1));assert((mint::mod()-1)%z==0);if(std::min(n,m)<=60)returnconvolution_naive(a,b);returninternal::convolution_fft(a,b);}template<classmint,internal::is_static_modint_t<mint>*=nullptr>std::vector<mint>convolution(conststd::vector<mint>&a,conststd::vector<mint>&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};intz=(int)internal::bit_ceil((unsignedint)(n+m-1));assert((mint::mod()-1)%z==0);if(std::min(n,m)<=60)returnconvolution_naive(a,b);returninternal::convolution_fft(a,b);}template<unsignedintmod=998244353,classT,std::enable_if_t<internal::is_integral<T>::value>*=nullptr>std::vector<T>convolution(conststd::vector<T>&a,conststd::vector<T>&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};usingmint=static_modint<mod>;intz=(int)internal::bit_ceil((unsignedint)(n+m-1));assert((mint::mod()-1)%z==0);std::vector<mint>a2(n),b2(m);for(inti=0;i<n;i++){a2[i]=mint(a[i]);}for(inti=0;i<m;i++){b2[i]=mint(b[i]);}autoc2=convolution(std::move(a2),std::move(b2));std::vector<T>c(n+m-1);for(inti=0;i<n+m-1;i++){c[i]=c2[i].val();}returnc;}std::vector<longlong>convolution_ll(conststd::vector<longlong>&a,conststd::vector<longlong>&b){intn=int(a.size()),m=int(b.size());if(!n||!m)return{};staticconstexprunsignedlonglongMOD1=754974721;// 2^24staticconstexprunsignedlonglongMOD2=167772161;// 2^25staticconstexprunsignedlonglongMOD3=469762049;// 2^26staticconstexprunsignedlonglongM2M3=MOD2*MOD3;staticconstexprunsignedlonglongM1M3=MOD1*MOD3;staticconstexprunsignedlonglongM1M2=MOD1*MOD2;staticconstexprunsignedlonglongM1M2M3=MOD1*MOD2*MOD3;staticconstexprunsignedlonglongi1=internal::inv_gcd(MOD2*MOD3,MOD1).second;staticconstexprunsignedlonglongi2=internal::inv_gcd(MOD1*MOD3,MOD2).second;staticconstexprunsignedlonglongi3=internal::inv_gcd(MOD1*MOD2,MOD3).second;staticconstexprintMAX_AB_BIT=24;static_assert(MOD1%(1ull<<MAX_AB_BIT)==1,"MOD1 isn't enough to support an array length of 2^24.");static_assert(MOD2%(1ull<<MAX_AB_BIT)==1,"MOD2 isn't enough to support an array length of 2^24.");static_assert(MOD3%(1ull<<MAX_AB_BIT)==1,"MOD3 isn't enough to support an array length of 2^24.");assert(n+m-1<=(1<<MAX_AB_BIT));autoc1=convolution<MOD1>(a,b);autoc2=convolution<MOD2>(a,b);autoc3=convolution<MOD3>(a,b);std::vector<longlong>c(n+m-1);for(inti=0;i<n+m-1;i++){unsignedlonglongx=0;x+=(c1[i]*i1)%MOD1*M2M3;x+=(c2[i]*i2)%MOD2*M1M3;x+=(c3[i]*i3)%MOD3*M1M2;// B = 2^63, -B <= x, r(real value) < B// (x, x - M, x - 2M, or x - 3M) = r (mod 2B)// r = c1[i] (mod MOD1)// focus on MOD1// r = x, x - M', x - 2M', x - 3M' (M' = M % 2^64) (mod 2B)// r = x,// x - M' + (0 or 2B),// x - 2M' + (0, 2B or 4B),// x - 3M' + (0, 2B, 4B or 6B) (without mod!)// (r - x) = 0, (0)// - M' + (0 or 2B), (1)// -2M' + (0 or 2B or 4B), (2)// -3M' + (0 or 2B or 4B or 6B) (3) (mod MOD1)// we checked that// ((1) mod MOD1) mod 5 = 2// ((2) mod MOD1) mod 5 = 3// ((3) mod MOD1) mod 5 = 4longlongdiff=c1[i]-internal::safe_mod((longlong)(x),(longlong)(MOD1));if(diff<0)diff+=MOD1;staticconstexprunsignedlonglongoffset[5]={0,0,M1M2M3,2*M1M2M3,3*M1M2M3};x-=offset[diff%5];c[i]=x;}returnc;}}// namespace e6nlaq