#include <bits/stdc++.h>
using namespace std;
using ll = long long;
const ll INFLL = (1LL<<60);
struct Node {
Node *l,*r;
int pr, sz;
ll val, mn, mx;
bool rev;
Node(ll v=0): l(nullptr), r(nullptr), pr((rand()<<16)^rand()), sz(1), val(v), mn(v), mx(v), rev(false) {}
};
int sz(Node* t){ return t? t->sz : 0; }
void pull(Node* t){ if(!t) return; t->sz = 1 + sz(t->l) + sz(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 push(Node* 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; }
Node* merge(Node* a, Node* b){
if(!a) return b; if(!b) return a;
push(a); push(b);
if(a->pr < b->pr){ a->r = merge(a->r, b); pull(a); return a; }
else { b->l = merge(a, b->l); pull(b); return b; }
}
void split(Node* t, int k, Node*& a, Node*& b){ // first k nodes -> a
if(!t){ a=b=nullptr; return; }
push(t);
int lsz = sz(t->l);
if(lsz >= k){
split(t->l, k, a, t->l);
pull(t);
b = t;
} else {
split(t->r, k - lsz - 1, t->r, b);
pull(t);
a = t;
}
}
// treap helpers
ll treap_range_min(Node*& root, int l, int r){
Node *a,*b,*c;
split(root, r, a, b);
split(a, l-1, c, a);
ll ans = (a? a->mn : INFLL);
a = merge(c, a);
root = merge(a, b);
return ans;
}
ll treap_range_max(Node*& root, int l, int r){
Node *a,*b,*c;
split(root, r, a, b);
split(a, l-1, c, a);
ll ans = (a? a->mx : -INFLL);
a = merge(c, a);
root = merge(a, b);
return ans;
}
// HLD
int N,Q;
vector<vector<int>> adj;
vector<int> parentv, depthv, heavy, head, pos, szv, invpos;
int curPos;
vector<ll> weightv;
int dfs1(int v,int p){
parentv[v]=p; depthv[v]=(p==-1?0:depthv[p]+1);
int size=1, maxc=0; heavy[v]=-1;
for(int to: adj[v]) if(to!=p){
int s = dfs1(to, v);
if(s>maxc){ maxc=s; heavy[v]=to; }
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]) 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;
}
int main(){
ios::sync_with_stdio(false);
cin.tie(nullptr);
srand(1234567);
if(!(cin>>N>>Q)) return 0;
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+1,0);
curPos = 1;
dfs1(1,-1);
dfs2(1,1);
Node* root = nullptr;
for(int i=1;i<=N;i++){
Node* nd = new Node(weightv[ invpos[i] ]);
root = merge(root, nd);
}
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, treap_range_min(root, 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, treap_range_max(root, s.l, s.r));
cout<<ans<<"\n";
} else {
auto segs_path = getPathSegments(u,v);
vector<Seg> segs_sorted = segs_path;
sort(segs_sorted.begin(), segs_sorted.end(), [](const Seg&a,const Seg&b){ return a.l < b.l; });
int m = segs_sorted.size();
vector<Node*> nonPath; nonPath.reserve(m+1);
vector<Node*> segsT; segsT.reserve(m);
Node* cur = root;
int prevR = 0; // <-- important: use prevR to index into current 'cur'
for(int i=0;i<m;i++){
int l = segs_sorted[i].l;
int r = segs_sorted[i].r;
int len = r - l + 1;
int kidx = l - prevR - 1; // CORRECT calculation
Node *left, *rest;
split(cur, kidx, left, rest);
Node *mid, *right;
split(rest, len, mid, right);
nonPath.push_back(left);
segsT.push_back(mid);
cur = right;
prevR = r;
}
nonPath.push_back(cur);
unordered_map<int,int> idx; idx.reserve(m*2+3);
for(int i=0;i<m;i++) idx[segs_sorted[i].l] = i;
Node* pathSeq = nullptr;
for(auto &s: segs_path){
int id = idx[s.l];
Node* t = segsT[id];
if(s.rev) { if(t) t->rev ^= 1; }
pathSeq = merge(pathSeq, t);
}
if(pathSeq) pathSeq->rev ^= 1;
Node* temp = pathSeq;
vector<Node*> newMidByL(m, nullptr);
for(int i=0;i<(int)segs_path.size();i++){
auto &s = segs_path[i];
int len = s.r - s.l + 1;
Node *left, *right;
split(temp, len, left, right);
if(s.rev){ if(left) left->rev ^= 1; }
int id = idx[s.l];
newMidByL[id] = left;
temp = right;
}
Node* res = nullptr;
for(int i=0;i<m;i++){
res = merge(res, nonPath[i]);
res = merge(res, newMidByL[i]);
}
res = merge(res, nonPath[m]);
root = res;
}
}
return 0;
}
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
const ll INFLL = (1LL<<60);

struct Node {
    Node *l,*r;
    int pr, sz;
    ll val, mn, mx;
    bool rev;
    Node(ll v=0): l(nullptr), r(nullptr), pr((rand()<<16)^rand()), sz(1), val(v), mn(v), mx(v), rev(false) {}
};
int sz(Node* t){ return t? t->sz : 0; }
void pull(Node* t){ if(!t) return; t->sz = 1 + sz(t->l) + sz(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 push(Node* 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; }

Node* merge(Node* a, Node* b){
    if(!a) return b; if(!b) return a;
    push(a); push(b);
    if(a->pr < b->pr){ a->r = merge(a->r, b); pull(a); return a; }
    else { b->l = merge(a, b->l); pull(b); return b; }
}
void split(Node* t, int k, Node*& a, Node*& b){ // first k nodes -> a
    if(!t){ a=b=nullptr; return; }
    push(t);
    int lsz = sz(t->l);
    if(lsz >= k){
        split(t->l, k, a, t->l);
        pull(t);
        b = t;
    } else {
        split(t->r, k - lsz - 1, t->r, b);
        pull(t);
        a = t;
    }
}

// treap helpers
ll treap_range_min(Node*& root, int l, int r){
    Node *a,*b,*c;
    split(root, r, a, b);
    split(a, l-1, c, a);
    ll ans = (a? a->mn : INFLL);
    a = merge(c, a);
    root = merge(a, b);
    return ans;
}
ll treap_range_max(Node*& root, int l, int r){
    Node *a,*b,*c;
    split(root, r, a, b);
    split(a, l-1, c, a);
    ll ans = (a? a->mx : -INFLL);
    a = merge(c, a);
    root = merge(a, b);
    return ans;
}

// HLD
int N,Q;
vector<vector<int>> adj;
vector<int> parentv, depthv, heavy, head, pos, szv, invpos;
int curPos;
vector<ll> weightv;

int dfs1(int v,int p){
    parentv[v]=p; depthv[v]=(p==-1?0:depthv[p]+1);
    int size=1, maxc=0; heavy[v]=-1;
    for(int to: adj[v]) if(to!=p){
        int s = dfs1(to, v);
        if(s>maxc){ maxc=s; heavy[v]=to; }
        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]) 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;
}

int main(){
    ios::sync_with_stdio(false);
    cin.tie(nullptr);
    srand(1234567);
    if(!(cin>>N>>Q)) return 0;
    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+1,0);
    curPos = 1;
    dfs1(1,-1);
    dfs2(1,1);
    Node* root = nullptr;
    for(int i=1;i<=N;i++){
        Node* nd = new Node(weightv[ invpos[i] ]);
        root = merge(root, nd);
    }

    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, treap_range_min(root, 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, treap_range_max(root, s.l, s.r));
            cout<<ans<<"\n";
        } else {
            auto segs_path = getPathSegments(u,v);
            vector<Seg> segs_sorted = segs_path;
            sort(segs_sorted.begin(), segs_sorted.end(), [](const Seg&a,const Seg&b){ return a.l < b.l; });
            int m = segs_sorted.size();
            vector<Node*> nonPath; nonPath.reserve(m+1);
            vector<Node*> segsT; segsT.reserve(m);
            Node* cur = root;
            int prevR = 0; // <-- important: use prevR to index into current 'cur'
            for(int i=0;i<m;i++){
                int l = segs_sorted[i].l;
                int r = segs_sorted[i].r;
                int len = r - l + 1;
                int kidx = l - prevR - 1; // CORRECT calculation
                Node *left, *rest;
                split(cur, kidx, left, rest);
                Node *mid, *right;
                split(rest, len, mid, right);
                nonPath.push_back(left);
                segsT.push_back(mid);
                cur = right;
                prevR = r;
            }
            nonPath.push_back(cur);
            unordered_map<int,int> idx; idx.reserve(m*2+3);
            for(int i=0;i<m;i++) idx[segs_sorted[i].l] = i;
            Node* pathSeq = nullptr;
            for(auto &s: segs_path){
                int id = idx[s.l];
                Node* t = segsT[id];
                if(s.rev) { if(t) t->rev ^= 1; }
                pathSeq = merge(pathSeq, t);
            }
            if(pathSeq) pathSeq->rev ^= 1;
            Node* temp = pathSeq;
            vector<Node*> newMidByL(m, nullptr);
            for(int i=0;i<(int)segs_path.size();i++){
                auto &s = segs_path[i];
                int len = s.r - s.l + 1;
                Node *left, *right;
                split(temp, len, left, right);
                if(s.rev){ if(left) left->rev ^= 1; }
                int id = idx[s.l];
                newMidByL[id] = left;
                temp = right;
            }
            Node* res = nullptr;
            for(int i=0;i<m;i++){
                res = merge(res, nonPath[i]);
                res = merge(res, newMidByL[i]);
            }
            res = merge(res, nonPath[m]);
            root = res;
        }
    }
    return 0;
}
