#define PROBLEM "https://yukicoder.me/problems/no/2588"
//#include"../../template/template.hpp"//#include"../../data-structure/union-find.hpp"
#include"../../segment-tree/segment-tree.hpp"
#include"../../tree/auxiliary-tree.hpp"
#include"../../tree/heavy-light-decomposition.hpp"
#include"../../tree/tree-query.hpp"//#include"../../modint/montgomery-modint.hpp"
#include"../../modulo/binomial.hpp"//usingnamespaceNyaan;usingmint=LazyMontgomeryModInt<998244353>;// using mint = LazyMontgomeryModInt<1000000007>;usingvm=vector<mint>;usingvvm=vector<vm>;Binomial<mint>C;usingnamespaceNyaan;voidq(){inl(N,M);autoG=graph(N,M);vvig(N);{vimx=mkiota(N);UnionFinduf(N);rep(i,N)each(j,G[i]){if(j<iand!uf.same(i,j)){intk=mx[uf.find(j)];g[i].push_back(k);g[k].push_back(i);uf.unite(i,k);mx[uf.find(i)]=i;}}}HeavyLightDecompositionhld(g,N-1);Treetree(g,N-1);SegmentTreedp(N,[](minta,mintb){returna+b;},mint{});AuxiliaryTreeauxgen(g,N-1);rep(i,N){mintcur=1;vichds;chds.push_back(i);each(j,G[i]){if(j<i)chds.push_back(j);}auto[aux,mp]=auxgen.get(chds);rep(ii,sz(aux))each(j,aux[ii]){intl=mp[ii],c=mp[j];intnxt=tree.nxt(l,c);hld.path_query(nxt,c,true,[&](intu,intv){cur+=dp.query(u,v);});}dp.update(hld.idx(i).fi,cur);}out(dp.query(0,N));}voidNyaan::solve(){intt=1;// in(t);while(t--)q();}
#line 1 "verify/verify-yuki/yuki-2588.test.cpp"
#define PROBLEM "https://yukicoder.me/problems/no/2588"
//#line 2 "template/template.hpp"
usingnamespacestd;// intrinstic#include<immintrin.h>#include<algorithm>
#include<array>
#include<bitset>
#include<cassert>
#include<cctype>
#include<cfenv>
#include<cfloat>
#include<chrono>
#include<cinttypes>
#include<climits>
#include<cmath>
#include<complex>
#include<cstdarg>
#include<cstddef>
#include<cstdint>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<deque>
#include<fstream>
#include<functional>
#include<initializer_list>
#include<iomanip>
#include<ios>
#include<iostream>
#include<istream>
#include<iterator>
#include<limits>
#include<list>
#include<map>
#include<memory>
#include<new>
#include<numeric>
#include<ostream>
#include<queue>
#include<random>
#include<set>
#include<sstream>
#include<stack>
#include<streambuf>
#include<string>
#include<tuple>
#include<type_traits>
#include<typeinfo>
#include<unordered_map>
#include<unordered_set>
#include<utility>
#include<vector>// utility#line 3 "template/util.hpp"
namespaceNyaan{usingll=longlong;usingi64=longlong;usingu64=unsignedlonglong;usingi128=__int128_t;usingu128=__uint128_t;template<typenameT>usingV=vector<T>;template<typenameT>usingVV=vector<vector<T>>;usingvi=vector<int>;usingvl=vector<longlong>;usingvd=V<double>;usingvs=V<string>;usingvvi=vector<vector<int>>;usingvvl=vector<vector<longlong>>;template<typenameT>usingminpq=priority_queue<T,vector<T>,greater<T>>;template<typenameT,typenameU>structP:pair<T,U>{template<typename...Args>P(Args...args):pair<T,U>(args...){}usingpair<T,U>::first;usingpair<T,U>::second;P&operator+=(constP&r){first+=r.first;second+=r.second;return*this;}P&operator-=(constP&r){first-=r.first;second-=r.second;return*this;}P&operator*=(constP&r){first*=r.first;second*=r.second;return*this;}template<typenameS>P&operator*=(constS&r){first*=r,second*=r;return*this;}Poperator+(constP&r)const{returnP(*this)+=r;}Poperator-(constP&r)const{returnP(*this)-=r;}Poperator*(constP&r)const{returnP(*this)*=r;}template<typenameS>Poperator*(constS&r)const{returnP(*this)*=r;}Poperator-()const{returnP{-first,-second};}};usingpl=P<ll,ll>;usingpi=P<int,int>;usingvp=V<pl>;constexprintinf=1001001001;constexprlonglonginfLL=4004004004004004004LL;template<typenameT>intsz(constT&t){returnt.size();}template<typenameT,typenameU>inlineboolamin(T&x,Uy){return(y<x)?(x=y,true):false;}template<typenameT,typenameU>inlineboolamax(T&x,Uy){return(x<y)?(x=y,true):false;}template<typenameT>inlineTMax(constvector<T>&v){return*max_element(begin(v),end(v));}template<typenameT>inlineTMin(constvector<T>&v){return*min_element(begin(v),end(v));}template<typenameT>inlinelonglongSum(constvector<T>&v){returnaccumulate(begin(v),end(v),0LL);}template<typenameT>intlb(constvector<T>&v,constT&a){returnlower_bound(begin(v),end(v),a)-begin(v);}template<typenameT>intub(constvector<T>&v,constT&a){returnupper_bound(begin(v),end(v),a)-begin(v);}constexprlonglongTEN(intn){longlongret=1,x=10;for(;n;x*=x,n>>=1)ret*=(n&1?x:1);returnret;}template<typenameT,typenameU>pair<T,U>mkp(constT&t,constU&u){returnmake_pair(t,u);}template<typenameT>vector<T>mkrui(constvector<T>&v,boolrev=false){vector<T>ret(v.size()+1);if(rev){for(inti=int(v.size())-1;i>=0;i--)ret[i]=v[i]+ret[i+1];}else{for(inti=0;i<int(v.size());i++)ret[i+1]=ret[i]+v[i];}returnret;};template<typenameT>vector<T>mkuni(constvector<T>&v){vector<T>ret(v);sort(ret.begin(),ret.end());ret.erase(unique(ret.begin(),ret.end()),ret.end());returnret;}template<typenameF>vector<int>mkord(intN,Ff){vector<int>ord(N);iota(begin(ord),end(ord),0);sort(begin(ord),end(ord),f);returnord;}template<typenameT>vector<int>mkinv(vector<T>&v){intmax_val=*max_element(begin(v),end(v));vector<int>inv(max_val+1,-1);for(inti=0;i<(int)v.size();i++)inv[v[i]]=i;returninv;}vector<int>mkiota(intn){vector<int>ret(n);iota(begin(ret),end(ret),0);returnret;}template<typenameT>Tmkrev(constT&v){Tw{v};reverse(begin(w),end(w));returnw;}template<typenameT>boolnxp(T&v){returnnext_permutation(begin(v),end(v));}// 返り値の型は入力の T に依存// i 要素目 : [0, a[i])template<typenameT>vector<vector<T>>product(constvector<T>&a){vector<vector<T>>ret;vector<T>v;autodfs=[&](autorc,inti)->void{if(i==(int)a.size()){ret.push_back(v);return;}for(intj=0;j<a[i];j++)v.push_back(j),rc(rc,i+1),v.pop_back();};dfs(dfs,0);returnret;}// F : void(T&), mod を取る操作// T : 整数型のときはオーバーフローに注意するtemplate<typenameT,typenameF>TPower(Ta,longlongn,constT&I,F&&f){static_assert(std::is_invocable_r_v<void,F&,T&>,"Power callback must be callable as void(T&)");Tres=I;for(;n;std::invoke(f,a=a*a),n>>=1){if(n&1)std::invoke(f,res=res*a);}returnres;}// T : 整数型のときはオーバーフローに注意するtemplate<typenameT>TPower(Ta,longlongn,constT&I=T{1}){autono_op=[](T&)->void{};returnPower(a,n,I,no_op);}template<typenameT>TRev(constT&v){Tres=v;reverse(begin(res),end(res));returnres;}template<typenameT>vector<T>Transpose(constvector<T>&v){usingU=typenameT::value_type;if(v.empty())return{};intH=v.size(),W=v[0].size();vectorres(W,T(H,U{}));for(inti=0;i<H;i++){for(intj=0;j<W;j++){res[j][i]=v[i][j];}}returnres;}template<typenameT>vector<T>Rotate(constvector<T>&v,intclockwise=true){usingU=typenameT::value_type;intH=v.size(),W=v[0].size();vectorres(W,T(H,U{}));for(inti=0;i<H;i++){for(intj=0;j<W;j++){if(clockwise){res[W-1-j][i]=v[i][j];}else{res[j][H-1-i]=v[i][j];}}}returnres;}}// namespace Nyaan#line 58 "template/template.hpp"
// bit operation#line 1 "template/bitop.hpp"
namespaceNyaan{__attribute__((target("popcnt")))inlineintpopcnt(constu64&a){return__builtin_popcountll(a);}inlineintlsb(constu64&a){returna?__builtin_ctzll(a):64;}inlineintctz(constu64&a){returna?__builtin_ctzll(a):64;}inlineintmsb(constu64&a){returna?63-__builtin_clzll(a):-1;}template<typenameT>inlineintgbit(constT&a,inti){return(a>>i)&1;}template<typenameT>inlinevoidsbit(T&a,inti,boolb){if(gbit(a,i)!=b)a^=T(1)<<i;}constexprlonglongPW(intn){return1LL<<n;}constexprlonglongMSK(intn){return(1LL<<n)-1;}}// namespace Nyaan#line 61 "template/template.hpp"
// inout#line 1 "template/inout.hpp"
namespaceNyaan{template<typenameT,typenameU>ostream&operator<<(ostream&os,constpair<T,U>&p){os<<p.first<<" "<<p.second;returnos;}template<typenameT,typenameU>istream&operator>>(istream&is,pair<T,U>&p){is>>p.first>>p.second;returnis;}template<typenameT>ostream&operator<<(ostream&os,constvector<T>&v){ints=(int)v.size();for(inti=0;i<s;i++)os<<(i?" ":"")<<v[i];returnos;}template<typenameT>istream&operator>>(istream&is,vector<T>&v){for(auto&x:v)is>>x;returnis;}istream&operator>>(istream&is,__int128_t&x){stringS;is>>S;x=0;intflag=0;for(auto&c:S){if(c=='-'){flag=true;continue;}x*=10;x+=c-'0';}if(flag)x=-x;returnis;}istream&operator>>(istream&is,__uint128_t&x){stringS;is>>S;x=0;for(auto&c:S){x*=10;x+=c-'0';}returnis;}ostream&operator<<(ostream&os,__int128_tx){if(x==0)returnos<<0;if(x<0)os<<'-',x=-x;stringS;while(x)S.push_back('0'+x%10),x/=10;reverse(begin(S),end(S));returnos<<S;}ostream&operator<<(ostream&os,__uint128_tx){if(x==0)returnos<<0;stringS;while(x)S.push_back('0'+x%10),x/=10;reverse(begin(S),end(S));returnos<<S;}voidin(){}template<typenameT,class...U>voidin(T&t,U&...u){cin>>t;in(u...);}voidout(){cout<<"\n";}template<typenameT,class...U,charsep=' '>voidout(constT&t,constU&...u){cout<<t;if(sizeof...(u))cout<<sep;out(u...);}structIoSetupNya{IoSetupNya(){cin.tie(nullptr);ios::sync_with_stdio(false);cout<<fixed<<setprecision(15);cerr<<fixed<<setprecision(7);}}iosetupnya;}// namespace Nyaan#line 64 "template/template.hpp"
// debug#line 1 "template/debug.hpp"
namespaceDebugImpl{template<typenameU,typename=void>structis_specialize:false_type{};template<typenameU>structis_specialize<U,typenameconditional<false,typenameU::iterator,void>::type>:true_type{};template<typenameU>structis_specialize<U,typenameconditional<false,decltype(U::first),void>::type>:true_type{};template<typenameU>structis_specialize<U,enable_if_t<is_integral<U>::value,void>>:true_type{};voiddump(constchar&t){cerr<<t;}voiddump(conststring&t){cerr<<t;}voiddump(constbool&t){cerr<<(t?"true":"false");}voiddump(__int128_tt){if(t==0)cerr<<0;if(t<0)cerr<<'-',t=-t;stringS;while(t)S.push_back('0'+t%10),t/=10;reverse(begin(S),end(S));cerr<<S;}voiddump(__uint128_tt){if(t==0)cerr<<0;stringS;while(t)S.push_back('0'+t%10),t/=10;reverse(begin(S),end(S));cerr<<S;}template<typenameU,enable_if_t<!is_specialize<U>::value,nullptr_t>=nullptr>voiddump(constU&t){cerr<<t;}template<typenameT>voiddump(constT&t,enable_if_t<is_integral<T>::value>*=nullptr){stringres;if(t==Nyaan::inf)res="inf";ifconstexpr(is_signed<T>::value){if(t==-Nyaan::inf)res="-inf";}ifconstexpr(sizeof(T)==8){if(t==Nyaan::infLL)res="inf";ifconstexpr(is_signed<T>::value){if(t==-Nyaan::infLL)res="-inf";}}if(res.empty())res=to_string(t);cerr<<res;}template<typenameT,typenameU>voiddump(constpair<T,U>&);template<typenameT>voiddump(constpair<T*,int>&);template<typenameT>voiddump(constT&t,enable_if_t<!is_void<typenameT::iterator>::value>*=nullptr){cerr<<"[ ";for(autoit=t.begin();it!=t.end();){dump(*it);cerr<<(++it==t.end()?"":", ");}cerr<<" ]";}template<typenameT,typenameU>voiddump(constpair<T,U>&t){cerr<<"( ";dump(t.first);cerr<<", ";dump(t.second);cerr<<" )";}template<typenameT>voiddump(constpair<T*,int>&t){cerr<<"[ ";for(inti=0;i<t.second;i++){dump(t.first[i]);cerr<<(i==t.second-1?"":", ");}cerr<<" ]";}voidtrace(){cerr<<endl;}template<typenameHead,typename...Tail>voidtrace(Head&&head,Tail&&...tail){cerr<<" ";dump(head);if(sizeof...(tail)!=0)cerr<<",";trace(std::forward<Tail>(tail)...);}}// namespace DebugImpl#ifdef NyaanDebug
#define trc(...) \
do { \
cerr << "## " << #__VA_ARGS__ << " = "; \
DebugImpl::trace(__VA_ARGS__); \
} while (0)
#else
#define trc(...) (void(0))
#endif
#ifdef NyaanLocal
#define trc2(...) \
do { \
cerr << "## " << #__VA_ARGS__ << " = "; \
DebugImpl::trace(__VA_ARGS__); \
} while (0)
#else
#define trc2(...) (void(0))
#endif
#line 67 "template/template.hpp"
// macro#line 1 "template/macro.hpp"
#define each(x, v) for (auto&& x : v)
#define each2(x, y, v) for (auto&& [x, y] : v)
#define all(v) (v).begin(), (v).end()
#define rep(i, N) for (long long i = 0; i < (long long)(N); i++)
#define repr(i, N) for (long long i = (long long)(N)-1; i >= 0; i--)
#define rep1(i, N) for (long long i = 1; i <= (long long)(N); i++)
#define repr1(i, N) for (long long i = (N); (long long)(i) > 0; i--)
#define reg(i, a, b) for (long long i = (a); i < (b); i++)
#define regr(i, a, b) for (long long i = (b)-1; i >= (a); i--)
#define fi first
#define se second
#define ini(...) \
int __VA_ARGS__; \
in(__VA_ARGS__)
#define inl(...) \
long long __VA_ARGS__; \
in(__VA_ARGS__)
#define ins(...) \
string __VA_ARGS__; \
in(__VA_ARGS__)
#define in2(s, t) \
for (int i = 0; i < (int)s.size(); i++) { \
in(s[i], t[i]); \
}
#define in3(s, t, u) \
for (int i = 0; i < (int)s.size(); i++) { \
in(s[i], t[i], u[i]); \
}
#define in4(s, t, u, v) \
for (int i = 0; i < (int)s.size(); i++) { \
in(s[i], t[i], u[i], v[i]); \
}
#define die(...) \
do { \
Nyaan::out(__VA_ARGS__); \
return; \
} while (0)
#line 70 "template/template.hpp"
namespaceNyaan{voidsolve();}intmain(){Nyaan::solve();}#line 4 "verify/verify-yuki/yuki-2588.test.cpp"
//#line 2 "data-structure/union-find.hpp"
structUnionFind{vector<int>data;UnionFind(intN):data(N,-1){}intfind(intk){returndata[k]<0?k:data[k]=find(data[k]);}intunite(intx,inty){if((x=find(x))==(y=find(y)))returnfalse;if(data[x]>data[y])swap(x,y);data[x]+=data[y];data[y]=x;returntrue;}// f(x, y) : x に y をマージtemplate<typenameF>intunite(intx,inty,constF&f){if((x=find(x))==(y=find(y)))returnfalse;if(data[x]>data[y])swap(x,y);data[x]+=data[y];data[y]=x;f(x,y);returntrue;}intsize(intk){return-data[find(k)];}intsame(intx,inty){returnfind(x)==find(y);}};/**
* @brief Union Find(Disjoint Set Union)
*/#line 2 "segment-tree/segment-tree.hpp"
template<typenameT,typenameF>structSegmentTree{intN;intsize;vector<T>seg;constFf;constTI;SegmentTree(F_f,constT&I_):N(0),size(0),f(_f),I(I_){}SegmentTree(int_N,F_f,constT&I_):f(_f),I(I_){init(_N);}SegmentTree(constvector<T>&v,F_f,TI_):f(_f),I(I_){init(v.size());for(inti=0;i<(int)v.size();i++){seg[i+size]=v[i];}build();}voidinit(int_N){N=_N;size=1;while(size<N)size<<=1;seg.assign(2*size,I);}voidset(intk,Tx){seg[k+size]=x;}voidbuild(){for(intk=size-1;k>0;k--){seg[k]=f(seg[2*k],seg[2*k+1]);}}voidupdate(intk,Tx){k+=size;seg[k]=x;while(k>>=1){seg[k]=f(seg[2*k],seg[2*k+1]);}}voidadd(intk,Tx){k+=size;seg[k]+=x;while(k>>=1){seg[k]=f(seg[2*k],seg[2*k+1]);}}// query to [a, b)Tquery(inta,intb){TL=I,R=I;for(a+=size,b+=size;a<b;a>>=1,b>>=1){if(a&1)L=f(L,seg[a++]);if(b&1)R=f(seg[--b],R);}returnf(L,R);}T&operator[](constint&k){returnseg[k+size];}// check(a[l] * ... * a[r-1]) が true となる最大の r// (右端まですべて true なら N を返す)template<classC>intmax_right(intl,Ccheck){assert(0<=l&&l<=N);assert(check(I)==true);if(l==N)returnN;l+=size;Tsm=I;do{while(l%2==0)l>>=1;if(!check(f(sm,seg[l]))){while(l<size){l=(2*l);if(check(f(sm,seg[l]))){sm=f(sm,seg[l]);l++;}}returnl-size;}sm=f(sm,seg[l]);l++;}while((l&-l)!=l);returnN;}// check(a[l] * ... * a[r-1]) が true となる最小の l// (左端まで true なら 0 を返す)template<typenameC>intmin_left(intr,Ccheck){assert(0<=r&&r<=N);assert(check(I)==true);if(r==0)return0;r+=size;Tsm=I;do{r--;while(r>1&&(r%2))r>>=1;if(!check(f(seg[r],sm))){while(r<size){r=(2*r+1);if(check(f(seg[r],sm))){sm=f(seg[r],sm);r--;}}returnr+1-size;}sm=f(seg[r],sm);}while((r&-r)!=r);return0;}};#line 2 "tree/auxiliary-tree.hpp"
#line 6 "tree/auxiliary-tree.hpp"
usingnamespacestd;#line 2 "tree/heavy-light-decomposition.hpp"
#line 2 "graph/graph-template.hpp"
template<typenameT>structedge{intsrc,to;Tcost;edge(int_to,T_cost):src(-1),to(_to),cost(_cost){}edge(int_src,int_to,T_cost):src(_src),to(_to),cost(_cost){}edge&operator=(constint&x){to=x;return*this;}operatorint()const{returnto;}};template<typenameT>usingEdges=vector<edge<T>>;template<typenameT>usingWeightedGraph=vector<Edges<T>>;usingUnweightedGraph=vector<vector<int>>;// Input of (Unweighted) GraphUnweightedGraphgraph(intN,intM=-1,boolis_directed=false,boolis_1origin=true){UnweightedGraphg(N);if(M==-1)M=N-1;for(int_=0;_<M;_++){intx,y;cin>>x>>y;if(is_1origin)x--,y--;g[x].push_back(y);if(!is_directed)g[y].push_back(x);}returng;}// Input of Weighted Graphtemplate<typenameT>WeightedGraph<T>wgraph(intN,intM=-1,boolis_directed=false,boolis_1origin=true){WeightedGraph<T>g(N);if(M==-1)M=N-1;for(int_=0;_<M;_++){intx,y;cin>>x>>y;Tc;cin>>c;if(is_1origin)x--,y--;g[x].emplace_back(x,y,c);if(!is_directed)g[y].emplace_back(y,x,c);}returng;}// Input of Edgestemplate<typenameT>Edges<T>esgraph([[maybe_unused]]intN,intM,intis_weighted=true,boolis_1origin=true){Edges<T>es;for(int_=0;_<M;_++){intx,y;cin>>x>>y;Tc;if(is_weighted)cin>>c;elsec=1;if(is_1origin)x--,y--;es.emplace_back(x,y,c);}returnes;}// Input of Adjacency Matrixtemplate<typenameT>vector<vector<T>>adjgraph(intN,intM,TINF,intis_weighted=true,boolis_directed=false,boolis_1origin=true){vector<vector<T>>d(N,vector<T>(N,INF));for(int_=0;_<M;_++){intx,y;cin>>x>>y;Tc;if(is_weighted)cin>>c;elsec=1;if(is_1origin)x--,y--;d[x][y]=c;if(!is_directed)d[y][x]=c;}returnd;}/**
* @brief グラフテンプレート
*/#line 4 "tree/heavy-light-decomposition.hpp"
template<typenameG>structHeavyLightDecomposition{private:voiddfs_sz(intcur){size[cur]=1;for(auto&dst:g[cur]){if(dst==par[cur]){if(g[cur].size()>=2&&int(dst)==int(g[cur][0]))swap(g[cur][0],g[cur][1]);elsecontinue;}depth[dst]=depth[cur]+1;par[dst]=cur;dfs_sz(dst);size[cur]+=size[dst];if(size[dst]>size[g[cur][0]]){swap(dst,g[cur][0]);}}}voiddfs_hld(intcur){down[cur]=id++;for(autodst:g[cur]){if(dst==par[cur])continue;nxt[dst]=(int(dst)==int(g[cur][0])?nxt[cur]:int(dst));dfs_hld(dst);}up[cur]=id;}// [u, v)vector<pair<int,int>>ascend(intu,intv)const{vector<pair<int,int>>res;while(nxt[u]!=nxt[v]){res.emplace_back(down[u],down[nxt[u]]);u=par[nxt[u]];}if(u!=v)res.emplace_back(down[u],down[v]+1);returnres;}// (u, v]vector<pair<int,int>>descend(intu,intv)const{if(u==v)return{};if(nxt[u]==nxt[v])return{{down[u]+1,down[v]}};autores=descend(u,par[nxt[v]]);res.emplace_back(down[nxt[v]],down[v]);returnres;}public:G&g;introot,id;vector<int>size,depth,down,up,nxt,par;HeavyLightDecomposition(G&_g,int_root=0):g(_g),root(_root),id(0),size(g.size(),0),depth(g.size(),0),down(g.size(),-1),up(g.size(),-1),nxt(g.size(),root),par(g.size(),root){dfs_sz(root);dfs_hld(root);}pair<int,int>idx(inti)const{returnmake_pair(down[i],up[i]);}template<typenameF>voidpath_query(intu,intv,boolvertex,constF&f){intl=lca(u,v);for(auto&&[a,b]:ascend(u,l)){ints=a+1,t=b;s>t?f(t,s):f(s,t);}if(vertex)f(down[l],down[l]+1);for(auto&&[a,b]:descend(l,v)){ints=a,t=b+1;s>t?f(t,s):f(s,t);}}template<typenameF>voidpath_noncommutative_query(intu,intv,boolvertex,constF&f){intl=lca(u,v);for(auto&&[a,b]:ascend(u,l))f(a+1,b);if(vertex)f(down[l],down[l]+1);for(auto&&[a,b]:descend(l,v))f(a,b+1);}template<typenameF>voidsubtree_query(intu,boolvertex,constF&f){f(down[u]+int(!vertex),up[u]);}intlca(inta,intb){while(nxt[a]!=nxt[b]){if(down[a]<down[b])swap(a,b);a=par[nxt[a]];}returndepth[a]<depth[b]?a:b;}intdist(inta,intb){returndepth[a]+depth[b]-depth[lca(a,b)]*2;}};/**
* @brief Heavy Light Decomposition(重軽分解)
*/#line 9 "tree/auxiliary-tree.hpp"
template<typenameG>structAuxiliaryTree{Gg;HeavyLightDecomposition<G>hld;AuxiliaryTree(constG&_g,introot=0):g(_g),hld(g,root){}// ps : 頂点集合// 返り値 : (aux tree, aux tree の頂点と g の頂点の対応表)// aux tree は 親->子 の向きの辺のみ含まれる// ps が空の場合は空のグラフを返すpair<vector<vector<int>>,vector<int>>get(vector<int>ps){if(ps.empty())return{};autocomp=[&](inti,intj){returnhld.down[i]<hld.down[j];};sort(begin(ps),end(ps),comp);for(inti=0,ie=ps.size();i+1<ie;i++){ps.push_back(hld.lca(ps[i],ps[i+1]));}sort(begin(ps),end(ps),comp);ps.erase(unique(begin(ps),end(ps)),end(ps));vector<vector<int>>aux(ps.size());vector<int>rs;rs.push_back(0);for(inti=1;i<(int)ps.size();i++){intl=hld.lca(ps[rs.back()],ps[i]);while(ps[rs.back()]!=l)rs.pop_back();aux[rs.back()].push_back(i);rs.push_back(i);}returnmake_pair(aux,ps);}};/**
* @brief Auxiliary Tree
*/#line 3 "tree/tree-query.hpp"
template<typenameG>structTree{private:G&g;introot;vector<array<int,24>>bl;vector<int>dp;voidbuild(){bl.resize(g.size());dp.resize(g.size());for(auto&v:bl)fill(begin(v),end(v),-1);dfs(root,-1,0);}voiddfs(intc,intp,int_dp){dp[c]=_dp;for(inti=p,x=0;i!=-1;){bl[c][x]=i;i=bl[i][x],x++;}for(auto&d:g[c]){if(d==p)continue;dfs(d,c,_dp+1);}}public:Tree(G&_g,int_r=0):g(_g),root(_r){build();}intdepth(intu)const{returndp[u];}intpar(intu)const{returnu==root?-1:bl[u][0];}intkth_ancestor(intu,intk)const{if(dp[u]<k)return-1;while(k){intt=__builtin_ctz(k);u=bl[u][t],k^=1<<t;}returnu;}intnxt(ints,intt)const{if(dp[s]>=dp[t])returnpar(s);intu=kth_ancestor(t,dp[t]-dp[s]-1);returnbl[u][0]==s?u:bl[s][0];}vector<int>path(ints,intt)const{vector<int>pre,suf;while(dp[s]>dp[t]){pre.push_back(s);s=bl[s][0];}while(dp[s]<dp[t]){suf.push_back(t);t=bl[t][0];}while(s!=t){pre.push_back(s);suf.push_back(t);s=bl[s][0];t=bl[t][0];}pre.push_back(s);reverse(begin(suf),end(suf));copy(begin(suf),end(suf),back_inserter(pre));returnpre;}intlca(intu,intv){if(dp[u]!=dp[v]){if(dp[u]>dp[v])swap(u,v);v=kth_ancestor(v,dp[v]-dp[u]);}if(u==v)returnu;for(inti=__lg(dp[u]);i>=0;--i){if(dp[u]<(1<<i))continue;if(bl[u][i]!=bl[v][i])u=bl[u][i],v=bl[v][i];}returnbl[u][0];}// u - v 間のパス上の頂点のうち u から距離 i の頂点// (dist(u, v) < i のとき -1)intjump(intu,intv,inti){intlc=lca(u,v);intd1=dp[u]-dp[lc];if(i<=d1)returnkth_ancestor(u,i);intd=d1+dp[v]-dp[lc];if(i<=d)returnkth_ancestor(v,d-i);return-1;}};/**
* @brief 木に対する一般的なクエリ
*/#line 10 "verify/verify-yuki/yuki-2588.test.cpp"
//#line 2 "modint/montgomery-modint.hpp"
#line 5 "modint/montgomery-modint.hpp"
template<uint32_tmod>structLazyMontgomeryModInt{usingmint=LazyMontgomeryModInt;usingi32=int32_t;usingu32=uint32_t;usingu64=uint64_t;staticconstexpru32get_r(){u32ret=mod;for(i32i=0;i<4;++i)ret*=2-mod*ret;returnret;}staticconstexpru32r=get_r();staticconstexpru32n2=-u64(mod)%mod;static_assert(mod<(1<<30),"invalid, mod >= 2 ^ 30");static_assert((mod&1)==1,"invalid, mod % 2 == 0");static_assert(r*mod==1,"this code has bugs.");u32a;constexprLazyMontgomeryModInt():a(0){}constexprLazyMontgomeryModInt(constint64_t&b):a(reduce(u64(b%mod+mod)*n2)){};staticconstexpru32reduce(constu64&b){return(b+u64(u32(b)*u32(-r))*mod)>>32;}constexprmint&operator+=(constmint&b){if(i32(a+=b.a-2*mod)<0)a+=2*mod;return*this;}constexprmint&operator-=(constmint&b){if(i32(a-=b.a)<0)a+=2*mod;return*this;}constexprmint&operator*=(constmint&b){a=reduce(u64(a)*b.a);return*this;}constexprmint&operator/=(constmint&b){*this*=b.inverse();return*this;}constexprmintoperator+(constmint&b)const{returnmint(*this)+=b;}constexprmintoperator-(constmint&b)const{returnmint(*this)-=b;}constexprmintoperator*(constmint&b)const{returnmint(*this)*=b;}constexprmintoperator/(constmint&b)const{returnmint(*this)/=b;}constexprbooloperator==(constmint&b)const{return(a>=mod?a-mod:a)==(b.a>=mod?b.a-mod:b.a);}constexprbooloperator!=(constmint&b)const{return(a>=mod?a-mod:a)!=(b.a>=mod?b.a-mod:b.a);}constexprmintoperator-()const{returnmint()-mint(*this);}constexprmintoperator+()const{returnmint(*this);}constexprmintpow(u64n)const{mintret(1),mul(*this);while(n>0){if(n&1)ret*=mul;mul*=mul;n>>=1;}returnret;}constexprmintinverse()const{intx=get(),y=mod,u=1,v=0,t=0,tmp=0;while(y>0){t=x/y;x-=t*y,u-=t*v;tmp=x,x=y,y=tmp;tmp=u,u=v,v=tmp;}returnmint{u};}friendstd::ostream&operator<<(std::ostream&os,constmint&b){returnos<<b.get();}friendstd::istream&operator>>(std::istream&is,mint&b){int64_tt;is>>t;b=LazyMontgomeryModInt<mod>(t);return(is);}constexpru32get()const{u32ret=reduce(a);returnret>=mod?ret-mod:ret;}staticconstexpru32get_mod(){returnmod;}};#line 2 "modulo/binomial.hpp"
#line 6 "modulo/binomial.hpp"
usingnamespacestd;// コンストラクタの MAX に 「C(n, r) や fac(n) でクエリを投げる最大の n 」// を入れると倍速くらいになる// mod を超えて前計算して 0 割りを踏むバグは対策済みtemplate<typenameT>structBinomial{vector<T>f,g,h;Binomial(intMAX=0){assert(T::get_mod()!=0&&"Binomial<mint>()");f.resize(1,T{1});g.resize(1,T{1});h.resize(1,T{1});if(MAX>0)extend(MAX+1);}voidextend(intm=-1){intn=f.size();if(m==-1)m=n*2;m=min<int>(m,T::get_mod());if(n>=m)return;f.resize(m);g.resize(m);h.resize(m);for(inti=n;i<m;i++)f[i]=f[i-1]*T(i);g[m-1]=f[m-1].inverse();h[m-1]=g[m-1]*f[m-2];for(inti=m-2;i>=n;i--){g[i]=g[i+1]*T(i+1);h[i]=g[i]*f[i-1];}}Tfac(inti){if(i<0)returnT(0);while(i>=(int)f.size())extend();returnf[i];}Tfinv(inti){if(i<0)returnT(0);while(i>=(int)g.size())extend();returng[i];}Tinv(inti){if(i<0)return-inv(-i);while(i>=(int)h.size())extend();returnh[i];}TC(intn,intr){if(n<0||n<r||r<0)returnT(0);returnfac(n)*finv(n-r)*finv(r);}inlineToperator()(intn,intr){returnC(n,r);}template<typenameI>Tmultinomial(constvector<I>&r){static_assert(is_integral<I>::value==true);intn=0;for(auto&x:r){if(x<0)returnT(0);n+=x;}Tres=fac(n);for(auto&x:r)res*=finv(x);returnres;}template<typenameI>Toperator()(constvector<I>&r){returnmultinomial(r);}TC_naive(intn,intr){if(n<0||n<r||r<0)returnT(0);Tret=T(1);r=min(r,n-r);for(inti=1;i<=r;++i)ret*=inv(i)*(n--);returnret;}TP(intn,intr){if(n<0||n<r||r<0)returnT(0);returnfac(n)*finv(n-r);}// [x^r] 1 / (1-x)^nTH(intn,intr){if(n<0||r<0)returnT(0);returnr==0?1:C(n+r-1,r);}};#line 13 "verify/verify-yuki/yuki-2588.test.cpp"
//usingnamespaceNyaan;usingmint=LazyMontgomeryModInt<998244353>;// using mint = LazyMontgomeryModInt<1000000007>;usingvm=vector<mint>;usingvvm=vector<vm>;Binomial<mint>C;usingnamespaceNyaan;voidq(){inl(N,M);autoG=graph(N,M);vvig(N);{vimx=mkiota(N);UnionFinduf(N);rep(i,N)each(j,G[i]){if(j<iand!uf.same(i,j)){intk=mx[uf.find(j)];g[i].push_back(k);g[k].push_back(i);uf.unite(i,k);mx[uf.find(i)]=i;}}}HeavyLightDecompositionhld(g,N-1);Treetree(g,N-1);SegmentTreedp(N,[](minta,mintb){returna+b;},mint{});AuxiliaryTreeauxgen(g,N-1);rep(i,N){mintcur=1;vichds;chds.push_back(i);each(j,G[i]){if(j<i)chds.push_back(j);}auto[aux,mp]=auxgen.get(chds);rep(ii,sz(aux))each(j,aux[ii]){intl=mp[ii],c=mp[j];intnxt=tree.nxt(l,c);hld.path_query(nxt,c,true,[&](intu,intv){cur+=dp.query(u,v);});}dp.update(hld.idx(i).fi,cur);}out(dp.query(0,N));}voidNyaan::solve(){intt=1;// in(t);while(t--)q();}