#include <iostream>
#include <bits/stdc++.h>
#include <stdio.h>
using namespace std;
typedef long long int ll;
vector<ll> adj[500001];
ll d[500001];//depth/level
ll it[500001];
ll tt[500001];
ll r[500001][2];//subtree[ii] range
ll l[500001];//leaf or not
vector<ll> qq[500001*2]; //a,b,time,sort_by,ans_order
ll co=-1;
ll cc,dd;
ll tree[2][500001*4];
ll lazy[2][500001*4];
/* si -> index of current node in segment tree[ii]
ss and se -> Starting and ending indexes of elements for
which current nodes stores sum.
us and ue -> starting and ending indexes of update query
diff -> which we need to add in the range us to ue */
ll n;
void update(ll si, ll ss,ll se,ll l, ll r, ll value,ll ii) {
for (l += n, r += n; l < r; l >>= 1, r >>= 1) {
if (l&1) tree[ii][l++] += value;
if (r&1) tree[ii][--r] += value;
}
}
void update2(ll si, ll ss, ll se, ll us,
ll ue, ll diff,ll ii)
{
// If lazy[ii] value is non-zero for current node of segment
// tree[ii], then there are some pending updates. So we need
// to make sure that the pending updates are done before
// making new updates. Because this value may be used by
// parent after recursive calls (See last line of this
// function)
if (lazy[ii][si] != 0)
{
// Make pending updates using value stored in lazy[ii]
// nodes
tree[ii][si] += (se-ss+1)*lazy[ii][si];
// checking if it is not leaf node because if
// it is leaf node then we cannot go further
if (ss != se)
{
// We can postpone updating children we don't
// need their new values now.
// Since we are not yet updating children of si,
// we need to set lazy[ii] flags for the children
lazy[ii][si*2 + 1] += lazy[ii][si];
lazy[ii][si*2 + 2] += lazy[ii][si];
}
// Set the lazy[ii] value for current node as 0 as it
// has been updated
lazy[ii][si] = 0;
}
// out of range
if (ss>se || ss>ue || se<us)
return ;
// Current segment is fully in range
if (ss>=us && se<=ue)
{
// Add the difference to current node
tree[ii][si] += (se-ss+1)*diff;
// same logic for checking leaf node or not
if (ss != se)
{
// This is where we store values in lazy[ii] nodes,
// rather than updating the segment tree[ii] itelf
// Since we don't need these updated values now
// we postpone updates by storing values in lazy[ii][]
lazy[ii][si*2 + 1] += diff;
lazy[ii][si*2 + 2] += diff;
}
return;
}
// If not completely in rang, but overlaps, recur for
// children,
ll mid = (ss+se)/2;
update(si*2+1, ss, mid, us, ue, diff,ii);
update(si*2+2, mid+1, se, us, ue, diff,ii);
// And use the result of children calls to update this
// node
tree[ii][si] = tree[ii][si*2+1] + tree[ii][si*2+2];
}
ll query(ll ss, ll se, ll p, ll qe, ll si,ll ii) {
ll res = 0;
for (p+=n; p> 0; p>>= 1) res += tree[ii][p];
return res;
}
ll query2(ll ss, ll se, ll qs, ll qe, ll si,ll ii)
{
// If lazy[ii] flag is set for current node of segment tree[ii],
// then there are some pending updates. So we need to
// make sure that the pending updates are done before
// processing the sub sum query
if (lazy[ii][si] != 0)
{
// Make pending updates to this node. Note that this
// node represents sum of elements in arr[ss..se] and
// all these elements must be increased by lazy[ii][si]
tree[ii][si] += (se-ss+1)*lazy[ii][si];
// checking if it is not leaf node because if
// it is leaf node then we cannot go further
if (ss != se)
{
// Since we are not yet updating children os si,
// we need to set lazy[ii] values for the children
lazy[ii][si*2+1] += lazy[ii][si];
lazy[ii][si*2+2] += lazy[ii][si];
}
// unset the lazy[ii] value for current node as it has
// been updated
lazy[ii][si] = 0;
}
// Out of range
if (ss>se || ss>qe || se<qs)
return 0;
// At this poll we are sure that pending lazy[ii] updates
// are done for current node. So we can return value
// (same as it was for query in our previous post)
// If this segment lies in range
if (ss>=qs && se<=qe)
return tree[ii][si];
// If a part of this segment overlaps with the given
// range
ll mid = (ss + se)/2;
return query(ss, mid, qs, qe, 2*si+1,ii) +
query(mid+1, se, qs, qe, 2*si+2,ii);
}
ll e[500001];
void build(ll ss, ll se, ll si,ll ii)
{
if (ss > se)
return ;
if (ss == se)
{
if(1==1){tree[ii][si] = 0;}
else{tree[ii][si]=it[e[ss]];}//it[ss];
return;
}
ll mid = (ss + se)/2;
build(ss, mid, si*2+1,ii);
build(mid+1, se, si*2+2,ii);
tree[ii][si] = tree[ii][si*2 + 1] + tree[ii][si*2 + 2];
}
ll sort_by(const vector<ll>& a,const vector<ll>& b){
if(a[3]!=b[3]){
return a[3]<b[3];
}
return a[1]!=-1 and b[1]==-1;
}
void dfs(ll no,ll par,ll lev){
ll st=0;
ll i;
co++;
d[no]=lev;
r[no][0]=co;
e[co]=no;
for(ll j=0;j<adj[no].size();j++){
i=adj[no][j];
if(i==par){
continue;
}
st=1;
//tt[i]+=it[no];
dfs(i,no,lev+1);
}
if(st==0){
// tt[no]+=it[no];
l[no]=1;
}
else{
l[no]=0;
}
r[no][1]=co;
}
int main() {
ios_base::sync_with_stdio(false);
cin.tie(NULL);
for(int i=0;i<2;i++){
for(int j=0;j<2;j++){
tree[i][j]=0;
}
}
ll t,k,i,j,tot;
ll q;
ll a,b;
cin>>n>>q;
for(ll i=0;i<n;i++){
adj[i].clear();
}
for(ll i=0;i<n-1;i++){
cin>>a>>b;
a-=1;
b-=1;
adj[a].push_back(b);
adj[b].push_back(a);
}
for(ll i=0;i<n;i++){
// scanf("%i",&it[i]);
cin>>it[i];
}
dfs(0,-1,1);
char s;
ll x=0;
for(ll i=0;i<n;i++){
qq[q+i].push_back(i);
qq[q+i].push_back(it[i]);
qq[q+i].push_back(-1);
qq[q+i].push_back(-1-d[i]);
}
for(ll _=0;_<q;_++){
cin>>s;
if(s=='?'){
cin>>a;
qq[_].push_back(a-1);
qq[_].push_back(-1);
qq[_].push_back(_);
qq[_].push_back(_-d[a-1]);
qq[_].push_back(x);
x++;
}
else{
cin>>a>>b;
qq[_].push_back(a-1);
qq[_].push_back(b);
qq[_].push_back(_);
qq[_].push_back(_-d[a-1]);
}
}
ll ans[x];
//now to divide queries groups with sane 4th index and then seg tree
sort(qq,qq+q+n,sort_by);
ll prev=n+10;
vector<ll> cur;
vector<vector<ll>> rem;
/* for(ll i=0;i<n;i++){
cout<<e[i]<<" ";
}
cout<<endl;*/
/* for(ll i=0;i<q;i++){
for(ll j=0;j<qq[i].size();j++){
cout<<qq[i][j]<<" ";
}
cout<<endl;
}
for(ll i=0;i<n;i++){
cout<<d[i]<<" ";
}
cout<<endl;*/
for(ll i=0;i<q+n;i++){
cur=qq[i];
if(cur[3]!=prev and i>0){
for(ll j=0;j<rem.size();j++){
update(0,0,n-1,r[rem[j][0]][0],r[rem[j][0]][1],-rem[j][1],0);
}
rem.clear();
}
if(cur.size()==4){
cout<<r[cur[0]][0]<<" "<<r[cur[0]][1]<<" "<<cur[1]<<endl;
update(0,0,n-1,r[cur[0]][0],r[cur[0]][1],cur[1],0);
update(0,0,n-1,r[cur[0]][0],r[cur[0]][1],cur[1],1);
rem.push_back(cur);
}
else{
cout<<query(0,n-1,r[cur[0]][0],r[cur[0]][0],0,l[cur[0]])<<" "<<r[cur[0]][0]<<" "<<r[cur[0]][1]<<endl;
// for(ll kk=0;kk<4*n;kk++){
// cout<<tree[l[cur[0]]][kk]<<",";
// }
// cout<<endl;
ans[cur[4]]=query(0,n-1,r[cur[0]][0],r[cur[0]][0],0,l[cur[0]]);
}
prev=cur[3];
}
for(ll i=0;i<x;i++){
cout<<ans[i]<<endl;
}
return 0;
}

#include <iostream>
#include <bits/stdc++.h>
#include <stdio.h>
using namespace std;
typedef long long int ll;
vector<ll> adj[500001];

ll d[500001];//depth/level
ll it[500001];
ll tt[500001]; 
ll r[500001][2];//subtree[ii] range
ll l[500001];//leaf or not


vector<ll> qq[500001*2];  //a,b,time,sort_by,ans_order
ll co=-1;
ll cc,dd;

ll tree[2][500001*4];
ll lazy[2][500001*4];

/*  si -> index of current node in segment tree[ii] 
    ss and se -> Starting and ending indexes of elements for 
                 which current nodes stores sum. 
    us and ue -> starting and ending indexes of update query 
    diff -> which we need to add in the range us to ue */
ll n;
void update(ll si, ll ss,ll se,ll l, ll r, ll value,ll ii) {
  for (l += n, r += n; l < r; l >>= 1, r >>= 1) {
    if (l&1) tree[ii][l++] += value;
    if (r&1) tree[ii][--r] += value;
  }
}

void update2(ll si, ll ss, ll se, ll us, 
                     ll ue, ll diff,ll ii) 
{ 
    // If lazy[ii] value is non-zero for current node of segment 
    // tree[ii], then there are some pending updates. So we need 
    // to make sure that the pending updates are done before 
    // making new updates. Because this value may be used by 
    // parent after recursive calls (See last line of this 
    // function) 
    if (lazy[ii][si] != 0) 
    { 
        // Make pending updates using value stored in lazy[ii] 
        // nodes 
        tree[ii][si] += (se-ss+1)*lazy[ii][si]; 
  
        // checking if it is not leaf node because if 
        // it is leaf node then we cannot go further 
        if (ss != se) 
        { 
            // We can postpone updating children we don't 
            // need their new values now. 
            // Since we are not yet updating children of si, 
            // we need to set lazy[ii] flags for the children 
            lazy[ii][si*2 + 1]   += lazy[ii][si]; 
            lazy[ii][si*2 + 2]   += lazy[ii][si]; 
        } 
  
        // Set the lazy[ii] value for current node as 0 as it 
        // has been updated 
        lazy[ii][si] = 0; 
    } 
  
    // out of range 
    if (ss>se || ss>ue || se<us) 
        return ; 
  
    // Current segment is fully in range 
    if (ss>=us && se<=ue) 
    { 
        // Add the difference to current node 
        tree[ii][si] += (se-ss+1)*diff; 
  
        // same logic for checking leaf node or not 
        if (ss != se) 
        { 
            // This is where we store values in lazy[ii] nodes, 
            // rather than updating the segment tree[ii] itelf 
            // Since we don't need these updated values now 
            // we postpone updates by storing values in lazy[ii][] 
            lazy[ii][si*2 + 1]   += diff; 
            lazy[ii][si*2 + 2]   += diff; 
        } 
        return; 
    } 
  
    // If not completely in rang, but overlaps, recur for 
    // children, 
    ll mid = (ss+se)/2; 
    update(si*2+1, ss, mid, us, ue, diff,ii); 
    update(si*2+2, mid+1, se, us, ue, diff,ii); 
  
    // And use the result of children calls to update this 
    // node 
    tree[ii][si] = tree[ii][si*2+1] + tree[ii][si*2+2]; 
} 

ll query(ll ss, ll se, ll p, ll qe, ll si,ll ii)  {
  ll res = 0;
  for (p+=n; p> 0;  p>>= 1) res += tree[ii][p];
  return res;
}
ll query2(ll ss, ll se, ll qs, ll qe, ll si,ll ii) 
{ 
    // If lazy[ii] flag is set for current node of segment tree[ii], 
    // then there are some pending updates. So we need to 
    // make sure that the pending updates are done before 
    // processing the sub sum query 
    if (lazy[ii][si] != 0) 
    { 
        // Make pending updates to this node. Note that this 
        // node represents sum of elements in arr[ss..se] and 
        // all these elements must be increased by lazy[ii][si] 
        tree[ii][si] += (se-ss+1)*lazy[ii][si]; 
  
        // checking if it is not leaf node because if 
        // it is leaf node then we cannot go further 
        if (ss != se) 
        { 
            // Since we are not yet updating children os si, 
            // we need to set lazy[ii] values for the children 
            lazy[ii][si*2+1] += lazy[ii][si]; 
            lazy[ii][si*2+2] += lazy[ii][si]; 
        } 
  
        // unset the lazy[ii] value for current node as it has 
        // been updated 
        lazy[ii][si] = 0; 
    } 
  
    // Out of range 
    if (ss>se || ss>qe || se<qs) 
        return 0; 
  
    // At this poll we are sure that pending lazy[ii] updates 
    // are done for current node. So we can return value  
    // (same as it was for query in our previous post) 
  
    // If this segment lies in range 
    if (ss>=qs && se<=qe) 
        return tree[ii][si]; 
  
    // If a part of this segment overlaps with the given 
    // range 
    ll mid = (ss + se)/2; 
    return query(ss, mid, qs, qe, 2*si+1,ii) + 
           query(mid+1, se, qs, qe, 2*si+2,ii); 
} 
  
 ll e[500001];
void build(ll ss, ll se, ll si,ll ii) 
{ 
    if (ss > se) 
        return ; 

    if (ss == se) 
    { 
        if(1==1){tree[ii][si] = 0;}
        else{tree[ii][si]=it[e[ss]];}//it[ss]; 
        return; 
    } 
    ll mid = (ss + se)/2; 
    build(ss, mid, si*2+1,ii); 
    build(mid+1, se, si*2+2,ii); 
  
    tree[ii][si] = tree[ii][si*2 + 1] + tree[ii][si*2 + 2]; 
} 
  
ll sort_by(const vector<ll>& a,const vector<ll>& b){

    if(a[3]!=b[3]){
        return a[3]<b[3];
    }
    return a[1]!=-1 and b[1]==-1;
}

void dfs(ll no,ll par,ll lev){
    ll st=0;
    ll i;
    co++;
    d[no]=lev;
    
    r[no][0]=co;
    e[co]=no;
    for(ll j=0;j<adj[no].size();j++){
        i=adj[no][j];
        if(i==par){
            continue;
        }
        st=1;
        //tt[i]+=it[no];
        dfs(i,no,lev+1);
    }
    if(st==0){
     //   tt[no]+=it[no];
        l[no]=1;
    }
    else{
        l[no]=0;
    }
    r[no][1]=co;
}

int main() {
    ios_base::sync_with_stdio(false);
    cin.tie(NULL);
    for(int i=0;i<2;i++){
		for(int j=0;j<2;j++){
			tree[i][j]=0;
		}
	}
    ll t,k,i,j,tot;
    ll q;
    ll a,b;
    cin>>n>>q;
    for(ll i=0;i<n;i++){
        adj[i].clear();
    }
    for(ll i=0;i<n-1;i++){
        cin>>a>>b;
        a-=1;
        b-=1;
        
        adj[a].push_back(b);
        adj[b].push_back(a);
    }

    for(ll i=0;i<n;i++){
    //	scanf("%i",&it[i]);
        cin>>it[i];
    }
    
    dfs(0,-1,1);
    
    char s; 
    ll x=0;
    for(ll i=0;i<n;i++){
    	qq[q+i].push_back(i);
    	qq[q+i].push_back(it[i]);
    	qq[q+i].push_back(-1);
    	qq[q+i].push_back(-1-d[i]);
    }
   
    for(ll _=0;_<q;_++){
        cin>>s;

        if(s=='?'){
            cin>>a;
            qq[_].push_back(a-1);
            qq[_].push_back(-1);
            qq[_].push_back(_);
            qq[_].push_back(_-d[a-1]);
            qq[_].push_back(x);
            x++;

        }
        else{
            cin>>a>>b;
            qq[_].push_back(a-1);
            qq[_].push_back(b);
            qq[_].push_back(_);
            qq[_].push_back(_-d[a-1]);
        }

    }
    ll ans[x];
//now to divide queries groups with sane 4th index and then seg tree
    sort(qq,qq+q+n,sort_by);
    ll prev=n+10;
    vector<ll> cur;
    vector<vector<ll>> rem;
 /*   for(ll i=0;i<n;i++){
    	cout<<e[i]<<" ";
    }
    cout<<endl;*/
  /*  for(ll i=0;i<q;i++){
    	for(ll j=0;j<qq[i].size();j++){
    		cout<<qq[i][j]<<" ";
    	}
    	cout<<endl;
    }
    for(ll i=0;i<n;i++){
    	cout<<d[i]<<" ";
    	
    }
    cout<<endl;*/
    
    for(ll i=0;i<q+n;i++){
        cur=qq[i];
        if(cur[3]!=prev and i>0){
            for(ll j=0;j<rem.size();j++){
                update(0,0,n-1,r[rem[j][0]][0],r[rem[j][0]][1],-rem[j][1],0);
                
            }
            rem.clear();
        }
       
        if(cur.size()==4){
        	cout<<r[cur[0]][0]<<" "<<r[cur[0]][1]<<" "<<cur[1]<<endl;
            update(0,0,n-1,r[cur[0]][0],r[cur[0]][1],cur[1],0);
            update(0,0,n-1,r[cur[0]][0],r[cur[0]][1],cur[1],1);
            rem.push_back(cur);
            
        }
        else{
        	cout<<query(0,n-1,r[cur[0]][0],r[cur[0]][0],0,l[cur[0]])<<" "<<r[cur[0]][0]<<" "<<r[cur[0]][1]<<endl;
        //    for(ll kk=0;kk<4*n;kk++){
        //    	cout<<tree[l[cur[0]]][kk]<<",";
        //    }
         //   cout<<endl;
            ans[cur[4]]=query(0,n-1,r[cur[0]][0],r[cur[0]][0],0,l[cur[0]]);
        }
        prev=cur[3];
        
    }
    
    for(ll i=0;i<x;i++){
        cout<<ans[i]<<endl;
    }
    
    
	return 0;
}
