#include <bits/stdc++.h>
using namespace std;
using ll = long long;
const ll INFLL = (1LL<<62);
struct Treap {
ll val, mn, mx;
int pr, sz;
bool rev;
Treap *l, *r;
Treap(ll v): val(v), mn(v), mx(v), pr(rand()), sz(1), rev(false), l(nullptr), r(nullptr) {}
};
int tsz(Treap* t){ return t? t->sz:0; }
void pull(Treap* t){
if(!t) return;
t->sz = 1 + tsz(t->l) + tsz(t->r);
t->mn = t->val;
t->mx = t->val;
if(t->l){ t->mn = min(t->mn, t->l->mn); t->mx = max(t->mx, t->l->mx); }
if(t->r){ t->mn = min(t->mn, t->r->mn); t->mx = max(t->mx, t->r->mx); }
}
void apply_rev(Treap* t){
if(!t) return;
t->rev ^= 1;
}
void push(Treap* t){
if(!t || !t->rev) return;
t->rev = false;
swap(t->l, t->r);
if(t->l) t->l->rev ^= 1;
if(t->r) t->r->rev ^= 1;
}
Treap* mergeTreap(Treap* a, Treap* b){
if(!a) return b;
if(!b) return a;
if(a->pr < b->pr){
push(a);
a->r = mergeTreap(a->r, b);
pull(a);
return a;
} else {
push(b);
b->l = mergeTreap(a, b->l);
pull(b);
return b;
}
}
void splitTreap(Treap* t, int k, Treap*& a, Treap*& b){
if(!t){ a=b=nullptr; return; }
push(t);
if(tsz(t->l) >= k){
splitTreap(t->l, k, a, t->l);
pull(t);
b = t;
} else {
splitTreap(t->r, k - tsz(t->l) - 1, t->r, b);
pull(t);
a = t;
}
}
int N,Q;
vector<vector<int>> adj;
vector<int> parentv, depthv, heavy, head, pos, szv, invpos;
int curPos=1;
vector<ll> weightv;
int dfs1(int v,int p){
parentv[v]=p;
depthv[v]= (p==-1?0:depthv[p]+1);
int size=1;
int maxc=0;
heavy[v]=-1;
for(int to: adj[v]){
if(to==p) continue;
int s = dfs1(to, v);
if(s>maxc){ maxc=s; heavy[v]=to; }
maxc = max(maxc, s);
size += s;
}
szv[v]=size;
return size;
}
void dfs2(int v,int h){
head[v]=h;
pos[v]=curPos;
invpos[curPos]=v;
curPos++;
if(heavy[v]!=-1) dfs2(heavy[v], h);
for(int to: adj[v]){
if(to==parentv[v] || to==heavy[v]) continue;
dfs2(to, to);
}
}
struct Seg { int l,r; bool rev; };
vector<Seg> getPathSegments(int u,int v){
vector<Seg> up, down;
while(head[u] != head[v]){
if(depthv[head[u]] >= depthv[head[v]]){
up.push_back({pos[head[u]], pos[u], true});
u = parentv[head[u]];
} else {
down.push_back({pos[head[v]], pos[v], false});
v = parentv[head[v]];
}
}
if(pos[u] <= pos[v]) up.push_back({pos[u], pos[v], false});
else up.push_back({pos[v], pos[u], true});
for(int i=(int)down.size()-1;i>=0;--i) up.push_back(down[i]);
return up;
}
Treap* root = nullptr;
ll range_min(int l,int r){
Treap *a,*b,*c;
splitTreap(root, r, a, b);
splitTreap(a, l-1, c, a);
ll ans = (a? a->mn : INFLL);
a = mergeTreap(c, a);
root = mergeTreap(a, b);
return ans;
}
ll range_max(int l,int r){
Treap *a,*b,*c;
splitTreap(root, r, a, b);
splitTreap(a, l-1, c, a);
ll ans = (a? a->mx : -INFLL);
a = mergeTreap(c, a);
root = mergeTreap(a, b);
return ans;
}
int main(){
ios::sync_with_stdio(false);
cin.tie(nullptr);
cin>>N>>Q;
weightv.assign(N+1,0);
for(int i=1;i<=N;i++) cin>>weightv[i];
adj.assign(N+1, {});
for(int i=0;i<N-1;i++){
int u,v; cin>>u>>v;
adj[u].push_back(v);
adj[v].push_back(u);
}
parentv.assign(N+1,-1);
depthv.assign(N+1,0);
heavy.assign(N+1,-1);
head.assign(N+1,0);
pos.assign(N+1,0);
szv.assign(N+1,0);
invpos.assign(N+2,0);
dfs1(1,-1);
dfs2(1,1);
root = nullptr;
for(int i=1;i<=N;i++){
Treap* node = new Treap(weightv[ invpos[i] ]);
root = mergeTreap(root, node);
}
while(Q--){
int type,u,v;
cin>>type>>u>>v;
if(type==1){
auto segs = getPathSegments(u,v);
ll ans = INFLL;
for(auto &s: segs){
ans = min(ans, range_min(s.l, s.r));
}
cout<<ans<<"\n";
} else if(type==2){
auto segs = getPathSegments(u,v);
ll ans = -INFLL;
for(auto &s: segs){
ans = max(ans, range_max(s.l, s.r));
}
cout<<ans<<"\n";
} else if(type==3){
auto pathSegs = getPathSegments(u,v);
int k = pathSegs.size();
vector<pair<int,int>> ranges;
ranges.reserve(k);
for(auto &s: pathSegs) ranges.push_back({s.l, s.r});
vector<pair<int,int>> uniq = ranges;
sort(uniq.begin(), uniq.end());
uniq.erase(unique(uniq.begin(), uniq.end()), uniq.end());
int m = uniq.size();
vector<Treap*> nonPath;
vector<Treap*> segsT;
Treap* cur = root;
int removed = 0;
for(int i=0;i<m;i++){
int l = uniq[i].first;
int r = uniq[i].second;
int len = r - l + 1;
int kidx = l - 1 - removed;
Treap *left, *rest;
splitTreap(cur, kidx, left, rest);
Treap *mid, *right;
splitTreap(rest, len, mid, right);
nonPath.push_back(left);
segsT.push_back(mid);
cur = right;
removed += len;
}
nonPath.push_back(cur);
unordered_map<int,int> idxmap;
for(int i=0;i<m;i++) idxmap[ uniq[i].first ] = i;
vector<Treap*> seqT;
seqT.reserve(k);
for(auto &ps: pathSegs){
int id = idxmap[ps.l];
Treap* tnode = segsT[id];
if(ps.rev) apply_rev(tnode);
seqT.push_back(tnode);
}
vector<Treap*> newSeq;
newSeq.resize(k);
for(int i=0;i<k;i++){
Treap* tmp = seqT[k-1 - i];
apply_rev(tmp);
newSeq[i] = tmp;
}
for(int i=0;i<k;i++){
auto &ps = pathSegs[i];
int id = idxmap[ps.l];
Treap* tnode = newSeq[i];
if(ps.rev) apply_rev(tnode);
segsT[id] = tnode;
}
Treap* res = nullptr;
for(int i=0;i<m;i++){
res = mergeTreap(res, nonPath[i]);
res = mergeTreap(res, segsT[i]);
}
res = mergeTreap(res, nonPath[m]);
root = res;
}
}
return 0;
}
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
const ll INFLL = (1LL<<62);
struct Treap {
    ll val, mn, mx;
    int pr, sz;
    bool rev;
    Treap *l, *r;
    Treap(ll v): val(v), mn(v), mx(v), pr(rand()), sz(1), rev(false), l(nullptr), r(nullptr) {}
};
int tsz(Treap* t){ return t? t->sz:0; }
void pull(Treap* t){
    if(!t) return;
    t->sz = 1 + tsz(t->l) + tsz(t->r);
    t->mn = t->val;
    t->mx = t->val;
    if(t->l){ t->mn = min(t->mn, t->l->mn); t->mx = max(t->mx, t->l->mx); }
    if(t->r){ t->mn = min(t->mn, t->r->mn); t->mx = max(t->mx, t->r->mx); }
}
void apply_rev(Treap* t){
    if(!t) return;
    t->rev ^= 1;
}
void push(Treap* t){
    if(!t || !t->rev) return;
    t->rev = false;
    swap(t->l, t->r);
    if(t->l) t->l->rev ^= 1;
    if(t->r) t->r->rev ^= 1;
}
Treap* mergeTreap(Treap* a, Treap* b){
    if(!a) return b;
    if(!b) return a;
    if(a->pr < b->pr){
        push(a);
        a->r = mergeTreap(a->r, b);
        pull(a);
        return a;
    } else {
        push(b);
        b->l = mergeTreap(a, b->l);
        pull(b);
        return b;
    }
}
void splitTreap(Treap* t, int k, Treap*& a, Treap*& b){
    if(!t){ a=b=nullptr; return; }
    push(t);
    if(tsz(t->l) >= k){
        splitTreap(t->l, k, a, t->l);
        pull(t);
        b = t;
    } else {
        splitTreap(t->r, k - tsz(t->l) - 1, t->r, b);
        pull(t);
        a = t;
    }
}
int N,Q;
vector<vector<int>> adj;
vector<int> parentv, depthv, heavy, head, pos, szv, invpos;
int curPos=1;
vector<ll> weightv;
int dfs1(int v,int p){
    parentv[v]=p;
    depthv[v]= (p==-1?0:depthv[p]+1);
    int size=1;
    int maxc=0;
    heavy[v]=-1;
    for(int to: adj[v]){
        if(to==p) continue;
        int s = dfs1(to, v);
        if(s>maxc){ maxc=s; heavy[v]=to; }
        maxc = max(maxc, s);
        size += s;
    }
    szv[v]=size;
    return size;
}
void dfs2(int v,int h){
    head[v]=h;
    pos[v]=curPos;
    invpos[curPos]=v;
    curPos++;
    if(heavy[v]!=-1) dfs2(heavy[v], h);
    for(int to: adj[v]){
        if(to==parentv[v] || to==heavy[v]) continue;
        dfs2(to, to);
    }
}
struct Seg { int l,r; bool rev; };
vector<Seg> getPathSegments(int u,int v){
    vector<Seg> up, down;
    while(head[u] != head[v]){
        if(depthv[head[u]] >= depthv[head[v]]){
            up.push_back({pos[head[u]], pos[u], true});
            u = parentv[head[u]];
        } else {
            down.push_back({pos[head[v]], pos[v], false});
            v = parentv[head[v]];
        }
    }
    if(pos[u] <= pos[v]) up.push_back({pos[u], pos[v], false});
    else up.push_back({pos[v], pos[u], true});
    for(int i=(int)down.size()-1;i>=0;--i) up.push_back(down[i]);
    return up;
}
Treap* root = nullptr;
ll range_min(int l,int r){
    Treap *a,*b,*c;
    splitTreap(root, r, a, b);
    splitTreap(a, l-1, c, a);
    ll ans = (a? a->mn : INFLL);
    a = mergeTreap(c, a);
    root = mergeTreap(a, b);
    return ans;
}
ll range_max(int l,int r){
    Treap *a,*b,*c;
    splitTreap(root, r, a, b);
    splitTreap(a, l-1, c, a);
    ll ans = (a? a->mx : -INFLL);
    a = mergeTreap(c, a);
    root = mergeTreap(a, b);
    return ans;
}
int main(){
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    cin>>N>>Q;
    weightv.assign(N+1,0);
    for(int i=1;i<=N;i++) cin>>weightv[i];
    adj.assign(N+1, {});
    for(int i=0;i<N-1;i++){
        int u,v; cin>>u>>v;
        adj[u].push_back(v);
        adj[v].push_back(u);
    }
    parentv.assign(N+1,-1);
    depthv.assign(N+1,0);
    heavy.assign(N+1,-1);
    head.assign(N+1,0);
    pos.assign(N+1,0);
    szv.assign(N+1,0);
    invpos.assign(N+2,0);
    dfs1(1,-1);
    dfs2(1,1);
    root = nullptr;
    for(int i=1;i<=N;i++){
        Treap* node = new Treap(weightv[ invpos[i] ]);
        root = mergeTreap(root, node);
    }
    while(Q--){
        int type,u,v;
        cin>>type>>u>>v;
        if(type==1){
            auto segs = getPathSegments(u,v);
            ll ans = INFLL;
            for(auto &s: segs){
                ans = min(ans, range_min(s.l, s.r));
            }
            cout<<ans<<"\n";
        } else if(type==2){
            auto segs = getPathSegments(u,v);
            ll ans = -INFLL;
            for(auto &s: segs){
                ans = max(ans, range_max(s.l, s.r));
            }
            cout<<ans<<"\n";
        } else if(type==3){
            auto pathSegs = getPathSegments(u,v);
            int k = pathSegs.size();
            vector<pair<int,int>> ranges;
            ranges.reserve(k);
            for(auto &s: pathSegs) ranges.push_back({s.l, s.r});
            vector<pair<int,int>> uniq = ranges;
            sort(uniq.begin(), uniq.end());
            uniq.erase(unique(uniq.begin(), uniq.end()), uniq.end());
            int m = uniq.size();
            vector<Treap*> nonPath;
            vector<Treap*> segsT;
            Treap* cur = root;
            int removed = 0;
            for(int i=0;i<m;i++){
                int l = uniq[i].first;
                int r = uniq[i].second;
                int len = r - l + 1;
                int kidx = l - 1 - removed;
                Treap *left, *rest;
                splitTreap(cur, kidx, left, rest);
                Treap *mid, *right;
                splitTreap(rest, len, mid, right);
                nonPath.push_back(left);
                segsT.push_back(mid);
                cur = right;
                removed += len;
            }
            nonPath.push_back(cur);
            unordered_map<int,int> idxmap;
            for(int i=0;i<m;i++) idxmap[ uniq[i].first ] = i;
            vector<Treap*> seqT;
            seqT.reserve(k);
            for(auto &ps: pathSegs){
                int id = idxmap[ps.l];
                Treap* tnode = segsT[id];
                if(ps.rev) apply_rev(tnode);
                seqT.push_back(tnode);
            }
            vector<Treap*> newSeq;
            newSeq.resize(k);
            for(int i=0;i<k;i++){
                Treap* tmp = seqT[k-1 - i];
                apply_rev(tmp);
                newSeq[i] = tmp;
            }
            for(int i=0;i<k;i++){
                auto &ps = pathSegs[i];
                int id = idxmap[ps.l];
                Treap* tnode = newSeq[i];
                if(ps.rev) apply_rev(tnode);
                segsT[id] = tnode;
            }
            Treap* res = nullptr;
            for(int i=0;i<m;i++){
                res = mergeTreap(res, nonPath[i]);
                res = mergeTreap(res, segsT[i]);
            }
            res = mergeTreap(res, nonPath[m]);
            root = res;
        }
    }
    return 0;
}
