#define PROBLEM "https://judge.yosupo.jp/problem/range_affine_range_sum"
#include<iostream>#include"data_structure/segtree/lazy_segtree.hpp"
#include"monoid/acted_monoids/range_affine_range_sum.hpp"#include"atcoder/modint.hpp"usingnamespacestd;usingnamespacem1une;usingmint=atcoder::modint998244353;intmain(){ios::sync_with_stdio(false);cin.tie(nullptr);intN,Q;cin>>N>>Q;lazy_segment_tree<range_affine_range_sum_monoid<mint>>seg(N);for(inti=0;i<N;i++){inta;cin>>a;seg.set(i,{mint(a),1});}while(Q--){intt;cin>>t;if(t==0){intl,r,b,c;cin>>l>>r>>b>>c;seg.apply(l,r,{mint(b),mint(c)});}else{intl,r;cin>>l>>r;mintans=seg.prod(l,r).value;cout<<ans.val()<<endl;}}}
#line 1 "verify/unit_test/lazy_segtree.test.cpp"
#define PROBLEM "https://judge.yosupo.jp/problem/range_affine_range_sum"
#include<iostream>#line 1 "data_structure/segtree/lazy_segtree.hpp"
#include<algorithm>
#include<functional>
#include<type_traits>
#include<vector>#line 1 "monoid/acted_monoid.hpp"
#include<concepts>
#line 7 "monoid/acted_monoid.hpp"
#line 1 "monoid/monoid.hpp"
#line 7 "monoid/monoid.hpp"
namespacem1une{template<typenameT,autooperation,autoidentity,boolcommutative>structmonoid{static_assert(std::is_invocable_r_v<T,decltype(operation),T,T>,"operation must work as T(T, T)");static_assert(std::is_invocable_r_v<T,decltype(identity)>,"identity must work as T()");usingvalue_type=T;staticconstexprautoop=operation;staticconstexprautoid=identity;staticconstexprboolis_commutative=commutative;};template<typenameT>conceptMonoid=requires(typenameT::value_typev){typenameT::value_type;{T::op(v,v)}->std::same_as<typenameT::value_type>;{T::id()}->std::same_as<typenameT::value_type>;{T::is_commutative}->std::convertible_to<bool>;};}// namespace m1une#line 9 "monoid/acted_monoid.hpp"
namespacem1une{template<MonoidData,MonoidAct,automapping>structacted_monoid{usingdata_monoid=Data;usingact_monoid=Act;usingdata_type=typenameData::value_type;usingact_type=typenameAct::value_type;static_assert(std::is_invocable_r_v<data_type,decltype(mapping),act_type,data_type>,"mapping must work as data_type(act_type, data_type)");staticconstexprautodata_op=Data::op;staticconstexprautodata_id=Data::id;staticconstexprbooldata_is_commutative=Data::is_commutative;staticconstexprautoact_op=Act::op;staticconstexprautoact_id=Act::id;staticconstexprboolact_is_commutative=Act::is_commutative;staticconstexprautoapply=mapping;};template<typenameT>conceptActedMonoid=requires(typenameT::data_typed,typenameT::act_typea){typenameT::data_monoid;typenameT::act_monoid;typenameT::data_type;typenameT::act_type;requiresMonoid<typenameT::data_monoid>;requiresMonoid<typenameT::act_monoid>;{T::apply(a,d)}->std::same_as<typenameT::data_type>;};}// namespace m1une#line 1 "utilities/bit_ceil.hpp"
namespacem1une{template<typenameT>constexprTbit_ceil(Tn){if(n<=1)return1;Tx=1;while(x<n)x<<=1;returnx;}}// namespace m1une#line 11 "data_structure/segtree/lazy_segtree.hpp"
namespacem1une{template<ActedMonoidAM>structlazy_segment_tree{usingS=typenameAM::data_type;usingF=typenameAM::act_type;private:int_n;int_size;int_log;std::vector<S>_data;std::vector<F>_lazy;voidupdate(intk){_data[k]=AM::data_op(_data[2*k],_data[2*k+1]);}voidall_apply(intk,Ff){_data[k]=AM::apply(f,_data[k]);if(k<_size){_lazy[k]=AM::act_op(_lazy[k],f);}}voidpush(intk){all_apply(2*k,_lazy[k]);all_apply(2*k+1,_lazy[k]);_lazy[k]=AM::act_id();}public:lazy_segment_tree():lazy_segment_tree(0){}explicitlazy_segment_tree(intn):lazy_segment_tree(std::vector<S>(n,AM::data_id())){}explicitlazy_segment_tree(conststd::vector<S>&v):_n(v.size()){_size=bit_ceil((unsignedint)_n);_log=0;while((1U<<_log)<(unsignedint)_size)_log++;_data.assign(2*_size,AM::data_id());_lazy.assign(_size,AM::act_id());for(inti=0;i<_n;i++){_data[_size+i]=v[i];}for(inti=_size-1;i>=1;i--){update(i);}}voidset(intp,Sx){p+=_size;for(inti=_log;i>=1;i--)push(p>>i);_data[p]=x;for(inti=1;i<=_log;i++)update(p>>i);}Sget(intp){p+=_size;for(inti=_log;i>=1;i--)push(p>>i);return_data[p];}Sprod(intl,intr){if(l==r)returnAM::data_id();l+=_size;r+=_size;for(inti=_log;i>=1;i--){if(((l>>i)<<i)!=l)push(l>>i);if(((r>>i)<<i)!=r)push((r-1)>>i);}Ssml=AM::data_id(),smr=AM::data_id();while(l<r){if(l&1)sml=AM::data_op(sml,_data[l++]);if(r&1)smr=AM::data_op(_data[--r],smr);l>>=1;r>>=1;}returnAM::data_op(sml,smr);}Sall_prod(){return_data[1];}voidapply(intp,Ff){p+=_size;for(inti=_log;i>=1;i--)push(p>>i);_data[p]=AM::apply(f,_data[p]);for(inti=1;i<=_log;i++)update(p>>i);}voidapply(intl,intr,Ff){if(l==r)return;l+=_size;r+=_size;for(inti=_log;i>=1;i--){if(((l>>i)<<i)!=l)push(l>>i);if(((r>>i)<<i)!=r)push((r-1)>>i);}{intl2=l,r2=r;while(l<r){if(l&1)all_apply(l++,f);if(r&1)all_apply(--r,f);l>>=1;r>>=1;}l=l2;r=r2;}for(inti=1;i<=_log;i++){if(((l>>i)<<i)!=l)update(l>>i);if(((r>>i)<<i)!=r)update((r-1)>>i);}}intmax_right(intl,autog){if(l==_n)return_n;l+=_size;for(inti=_log;i>=1;i--)push(l>>i);Ssm=AM::data_id();do{while(l%2==0)l>>=1;if(!g(AM::data_op(sm,_data[l]))){while(l<_size){push(l);l=(2*l);if(g(AM::data_op(sm,_data[l]))){sm=AM::data_op(sm,_data[l]);l++;}}returnl-_size;}sm=AM::data_op(sm,_data[l]);l++;}while((l&-l)!=l);return_n;}intmin_left(intr,autog){if(r==0)return0;r+=_size;for(inti=_log;i>=1;i--)push((r-1)>>i);Ssm=AM::data_id();do{r--;while(r>1&&(r%2))r>>=1;if(!g(AM::data_op(_data[r],sm))){while(r<_size){push(r);r=(2*r+1);if(g(AM::data_op(_data[r],sm))){sm=AM::data_op(_data[r],sm);r--;}}returnr+1-_size;}sm=AM::data_op(_data[r],sm);}while((r&-r)!=r);return0;}};}// namespace m1une#line 1 "monoid/acted_monoids/range_affine_range_sum.hpp"
#line 1 "monoid/monoid_addsz.hpp"
#line 5 "monoid/monoid_addsz.hpp"
#line 7 "monoid/monoid_addsz.hpp"
namespacem1une{template<typenameT>structvalue_and_size{Tvalue;intsize;};template<MonoidM>usingmonoid_addsz=monoid<value_and_size<typenameM::value_type>,[](value_and_size<typenameM::value_type>a,value_and_size<typenameM::value_type>b){returnvalue_and_size<typenameM::value_type>{M::op(a.value,b.value),a.size+b.size};},[](){returnvalue_and_size<typenameM::value_type>{M::id(),0};},M::is_commutative>;}// namespace m1une#line 1 "monoid/monoids/add_monoid.hpp"
#line 5 "monoid/monoids/add_monoid.hpp"
namespacem1une{template<typenameT>usingadd_monoid=monoid<T,[](Ta,Tb){returna+b;},[](){returnT(0);},true>;}// namespace m1une#line 1 "monoid/monoids/affine_monoid.hpp"
#include<utility>#line 7 "monoid/monoids/affine_monoid.hpp"
namespacem1une{// Affine transformation f(x) = ax + b is represented as (a, b)// perform f first, then g// op(f, g)(x) = g(f(x))template<typenameT>usingaffine_monoid=monoid<std::pair<T,T>,[](std::pair<T,T>f,std::pair<T,T>g){returnstd::pair<T,T>(f.first*g.first,f.second*g.first+g.second);},[](){returnstd::pair<T,T>(1,0);},false>;}// namespace m1une#line 8 "monoid/acted_monoids/range_affine_range_sum.hpp"
namespacem1une{template<typenameT>usingrange_affine_range_sum_monoid=acted_monoid<monoid_addsz<add_monoid<T>>,affine_monoid<T>,[](typenameaffine_monoid<T>::value_typef,typenamemonoid_addsz<add_monoid<T>>::value_typex){// f = (a, b) is the affine transformation// x = {value, size} is the data (sum and size of the range)// Applying f to each element xi and summing up:// sum(a*xi + b) = a * sum(xi) + b * sizereturntypenamemonoid_addsz<add_monoid<T>>::value_type{f.first*x.value+f.second*x.size,x.size};}>;}// namespace m1une#line 7 "verify/unit_test/lazy_segtree.test.cpp"
#line 1 "atcoder/modint.hpp"
#include<cassert>
#include<numeric>
#line 7 "atcoder/modint.hpp"
#ifdef _MSC_VER
#include<intrin.h>
#endif
#line 1 "atcoder/internal_math.hpp"
#line 5 "atcoder/internal_math.hpp"
#ifdef _MSC_VER
#include<intrin.h>
#endif
namespaceatcoder{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 atcoder#line 1 "atcoder/internal_type_traits.hpp"
#line 7 "atcoder/internal_type_traits.hpp"
namespaceatcoder{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 atcoder#line 14 "atcoder/modint.hpp"
namespaceatcoder{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());}intval()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());}intval()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 atcoder#line 9 "verify/unit_test/lazy_segtree.test.cpp"
usingnamespacestd;usingnamespacem1une;usingmint=atcoder::modint998244353;intmain(){ios::sync_with_stdio(false);cin.tie(nullptr);intN,Q;cin>>N>>Q;lazy_segment_tree<range_affine_range_sum_monoid<mint>>seg(N);for(inti=0;i<N;i++){inta;cin>>a;seg.set(i,{mint(a),1});}while(Q--){intt;cin>>t;if(t==0){intl,r,b,c;cin>>l>>r>>b>>c;seg.apply(l,r,{mint(b),mint(c)});}else{intl,r;cin>>l>>r;mintans=seg.prod(l,r).value;cout<<ans.val()<<endl;}}}