import java.io.IOException;
import java.io.InputStream;
import java.util.*;
class TestClass {
static final class InputReader {
private final byte[] buf = new byte[1024];
private int curChar;
private int numChars;
this.stream = stream;
}
if (curChar >= numChars) {
curChar = 0;
numChars = stream.read(buf);
if (numChars <= 0) {
return -1;
}
}
return buf[curChar++];
}
return (int) readLong();
}
int c = read();
while (isSpaceChar(c)) {
c = read();
}
boolean negative = false;
if (c == '-') {
negative = true;
c = read();
}
long res = 0;
do {
res *= 10;
res += c - '0';
c = read();
} while (!isSpaceChar(c));
return negative ? -res : res;
}
public final int[] readIntArray
(int size
) throws IOException { int[] array = new int[size];
for (int i = 0; i < size; i++) {
array[i] = readInt();
}
return array;
}
public final long[] readLongArray
(int size
) throws IOException { long[] array = new long[size];
for (int i = 0; i < size; i++) {
array[i] = readLong();
}
return array;
}
int c = read();
while (isSpaceChar(c)) {
c = read();
}
StringBuilder res = new StringBuilder();
do {
res.append((char) c);
c = read();
} while (!isSpaceChar(c));
return res.toString();
}
private boolean isSpaceChar(int c) {
return c == ' ' || c == '\n' || c == '\r' || c == '\t' || c == -1;
}
}
static long mulmod(long a, long b,
long mod) {
long res = 0; // Initialize result
a = a % mod;
while (b > 0) {
// If b is odd, add 'a' to result
if (b % 2 == 1) {
res = (res + a) % mod;
}
// Multiply 'a' with 2
a = (a * 2) % mod;
// Divide b by 2
b /= 2;
}
// Return result
return res % mod;
}
static long pow(long a, long b, long MOD) {
long x = 1, y = a;
while (b > 0) {
if (b % 2 == 1) {
x = (x * y);
if (x > MOD) x %= MOD;
}
y = (y * y);
if (y > MOD) y %= MOD;
b /= 2;
}
return x;
}
static long[] f = new long[100001];
static long InverseEuler(long n, long MOD) {
return pow(n, MOD - 2, MOD);
}
static long C(int n, int r, long MOD) {
return (f[n] * ((InverseEuler(f[r], MOD) * InverseEuler(f[n - r], MOD)) % MOD)) % MOD;
}
public static class SegmentTree {
long[] tree;
long[] lazy;
int n;
public SegmentTree(long[] arr) {
n = arr.length;
tree = new long[arr.length * 5];
lazy = new long[arr.length * 5];
build(arr, 0, arr.length - 1, 0);
}
private void build(long[] arr, int s, int e, int pos) {
if (s == e) {
tree[pos] = arr[s];
return;
}
int m = (s + e) / 2;
build(arr, s, m, 2 * pos + 1);
build(arr, m + 1, e, 2 * pos + 2);
tree
[pos
] = Math.
max(tree
[2 * pos
+ 1], tree
[2 * pos
+ 2]); }
public void update(int s, int e, long val) {
updateUtil(s, e, val, 0, n - 1, 0);
}
public long get(int s, int e) {
return getUtil(s, e, 0, n - 1, 0);
}
private long getUtil(int gs, int ge, int s, int e, int pos) {
if (s
> e
|| s
> ge
|| e
< gs
) return Long.
MIN_VALUE; if (lazy[pos] != 0) {
tree[pos] += lazy[pos];
if (s != e) {
lazy[2 * pos + 1] += lazy[pos];
lazy[2 * pos + 2] += lazy[pos];
}
lazy[pos] = 0;
}
if (s >= gs && e <= ge) {
return tree[pos];
}
int m = (s + e) / 2;
return Math.
max(getUtil
(gs, ge, s, m,
2 * pos
+ 1), getUtil
(gs, ge, m
+ 1, e,
2 * pos
+ 2)); }
private void updateUtil(int us, int ue, long val, int s, int e, int pos) {
if (s > e || s > ue || e < us) return;
if (lazy[pos] != 0) {
tree[pos] += lazy[pos];
if (s != e) {
lazy[2 * pos + 1] += lazy[pos];
lazy[2 * pos + 2] += lazy[pos];
}
lazy[pos] = 0;
}
if (s >= us && e <= ue) {
tree[pos] += val;
if (s != e) {
lazy[2 * pos + 1] += val;
lazy[2 * pos + 2] += val;
}
return;
}
int m = (s + e) / 2;
updateUtil(us, ue, val, s, m, 2 * pos + 1);
updateUtil(us, ue, val, m + 1, e, 2 * pos + 2);
tree
[pos
] = Math.
max(tree
[2 * pos
+ 1], tree
[2 * pos
+ 2]); }
}
static int[] h = {0, 0, -1, 1};
static int[] v = {1, -1, 0, 0};
public static class Pair {
public int x, y, p;
public Pair(int d, int y) {
this.x = d;
this.y = y;
}
}
static long compute_hash
(String s
) { int p = 31;
int m = 1000000007;
long hash_value = 0;
long p_pow = 1;
for (int i = 0; i < s.length(); ++i) {
char c = s.charAt(i);
hash_value = (hash_value + (c - 'a' + 1) * p_pow) % m;
p_pow = (p_pow * p) % m;
}
return hash_value;
}
static int counter = 0;
//https://i...content-available-to-author-only...e.com/ebRGa6
InputReader in
= new InputReader
(System.
in); long t = in.readLong();
int[][] pre = new int[100000+1][17];
int[] tin = new int[100000+1];
int[] tout = new int[100000+1];
int[] xors = new int[100000+1];
while (t-- > 0) {
counter = 0;
int n = in.readInt();
int q = in.readInt();
int[] val = new int[n+1];
for (int i = 1; i <= n; ++i) {
val[i] = in.readInt();
}
List<List<Integer>> tree = new ArrayList<>();
for (int i = 0; i <= n; ++i) {
tree.add(new ArrayList<>());
}
for (int i = 0; i < n - 1; ++i) {
int p = in.readInt();
int r = in.readInt();
tree.get(p).add(r);
tree.get(r ).add(p );
}
preprocess(1, pre, tree, new int[n+1], 0, -1, tin, tout, 0, val, xors);
// for (int i = 0; i < n; ++i) {
// for (int j = 30; j >= 0; --j) {
// System.out.print(pre[i][j] + " ");
// }
// System.out.println("");
// }
while (q-- > 0) {
int s = in.readInt();
int e = in.readInt();
if (e > s) {
int temp = s;
s = e;
e = temp;
}
if (s == e) {
continue;
}
if ((tin[s] <= tin[e] && tout[s] >= tout[e])) {
System.
out.
println(xors
[s
] ^ xors
[e
] ^ val
[s
]); continue;
}
if ((tin[s] >= tin[e] && tout[s] <= tout[e])) {
System.
out.
println(xors
[s
] ^ xors
[e
] ^ val
[e
]); continue;
}
int xor = xors[s];
for (int i = 16; i >= 0; --i) {
if (pre[s][i] != 0) {
if (!(tin[pre[s][i]] <= tin[e] && tout[pre[s][i]] >= tout[e])) {
s = pre[s][i];
}
}
}
// assert(pre[s][0] != -1);
xor = xor ^ xors[e] ^ val[pre[s][0]];
}
}
}
private static void preprocess(int pos, int[][] pre, List<List<Integer>> tree, int[] traverse, int depth, int last, int[] tin, int[] tout, int xor, int[] val, int[] xors) {
tin[pos] = counter++;
xors[pos] = xor ^ val[pos];
traverse[depth] = pos;
for (int i = 0; depth - (1 << i) >= 0; ++i) {
pre[pos][i] = traverse[depth - (1 << i)];
}
for (int i = 0; i < tree.get(pos).size(); ++i) {
if (tree.get(pos).get(i) != last)
preprocess(tree.get(pos).get(i), pre, tree, traverse, depth + 1, pos, tin, tout, xors[pos], val, xors);
}
tout[pos] = counter++;
}
private static long solve(long[][] dp, int n, int r) {
if (n == 0) {
if (r == 0) return 1;
else return 0;
}
if (r <= 0) return 0;
if (dp[n][r] != -1) return dp[n][r];
long answer = 0;
for (int i = 1; i < n; ++i) {
answer += solve(dp, n - 1, r - i);
}
dp[n][r] = answer;
return answer;
}
static int gcd(int a, int b) {
while (b != 0) {
int t = a;
a = b;
b = t % b;
}
return a;
}
class Solution {
public int maxProduct(int[] nums) {
PriorityQueue<Integer> p = new PriorityQueue<>(new Comparator<Integer>() {
@Override
return Integer.
compare(t1, integer
); }
});
for (int i = 0; i < nums.length; ++i) {
p.add(nums[i]);
}
int f = p.remove();
int s = p.remove();
return (f - 1) * (s - 1);
}
}
static boolean submit = true;
if (!submit)
}
static void debug(int s) {
if (!submit)
}
}


import java.io.IOException;
import java.io.InputStream;
import java.util.*;

class TestClass {


    static final class InputReader {
        private final InputStream stream;
        private final byte[] buf = new byte[1024];
        private int curChar;
        private int numChars;

        public InputReader(InputStream stream) {
            this.stream = stream;
        }

        private int read() throws IOException {
            if (curChar >= numChars) {
                curChar = 0;
                numChars = stream.read(buf);
                if (numChars <= 0) {
                    return -1;
                }
            }
            return buf[curChar++];
        }

        public final int readInt() throws IOException {
            return (int) readLong();
        }

        public final long readLong() throws IOException {
            int c = read();
            while (isSpaceChar(c)) {
                c = read();
                if (c == -1) throw new IOException();
            }
            boolean negative = false;
            if (c == '-') {
                negative = true;
                c = read();
            }
            long res = 0;
            do {
                res *= 10;
                res += c - '0';
                c = read();
            } while (!isSpaceChar(c));
            return negative ? -res : res;
        }

        public final int[] readIntArray(int size) throws IOException {
            int[] array = new int[size];
            for (int i = 0; i < size; i++) {
                array[i] = readInt();
            }
            return array;
        }

        public final long[] readLongArray(int size) throws IOException {
            long[] array = new long[size];
            for (int i = 0; i < size; i++) {
                array[i] = readLong();
            }
            return array;
        }

        public final String readString() throws IOException {
            int c = read();
            while (isSpaceChar(c)) {
                c = read();
            }
            StringBuilder res = new StringBuilder();
            do {
                res.append((char) c);
                c = read();
            } while (!isSpaceChar(c));
            return res.toString();
        }

        private boolean isSpaceChar(int c) {
            return c == ' ' || c == '\n' || c == '\r' || c == '\t' || c == -1;
        }
    }

    static long mulmod(long a, long b,
                       long mod) {
        long res = 0; // Initialize result
        a = a % mod;
        while (b > 0) {
            // If b is odd, add 'a' to result
            if (b % 2 == 1) {
                res = (res + a) % mod;
            }

            // Multiply 'a' with 2
            a = (a * 2) % mod;

            // Divide b by 2
            b /= 2;
        }

        // Return result
        return res % mod;
    }

    static long pow(long a, long b, long MOD) {
        long x = 1, y = a;
        while (b > 0) {
            if (b % 2 == 1) {
                x = (x * y);
                if (x > MOD) x %= MOD;
            }
            y = (y * y);
            if (y > MOD) y %= MOD;
            b /= 2;
        }
        return x;
    }

    static long[] f = new long[100001];

    static long InverseEuler(long n, long MOD) {
        return pow(n, MOD - 2, MOD);
    }

    static long C(int n, int r, long MOD) {

        return (f[n] * ((InverseEuler(f[r], MOD) * InverseEuler(f[n - r], MOD)) % MOD)) % MOD;
    }

    public static class SegmentTree {
        long[] tree;
        long[] lazy;
        int n;

        public SegmentTree(long[] arr) {
            n = arr.length;
            tree = new long[arr.length * 5];
            lazy = new long[arr.length * 5];
            build(arr, 0, arr.length - 1, 0);
        }

        private void build(long[] arr, int s, int e, int pos) {
            if (s == e) {
                tree[pos] = arr[s];
                return;
            }
            int m = (s + e) / 2;

            build(arr, s, m, 2 * pos + 1);
            build(arr, m + 1, e, 2 * pos + 2);

            tree[pos] = Math.max(tree[2 * pos + 1], tree[2 * pos + 2]);
        }

        public void update(int s, int e, long val) {
            updateUtil(s, e, val, 0, n - 1, 0);
        }

        public long get(int s, int e) {
            return getUtil(s, e, 0, n - 1, 0);
        }

        private long getUtil(int gs, int ge, int s, int e, int pos) {
            if (s > e || s > ge || e < gs) return Long.MIN_VALUE;
            if (lazy[pos] != 0) {
                tree[pos] += lazy[pos];
                if (s != e) {
                    lazy[2 * pos + 1] += lazy[pos];
                    lazy[2 * pos + 2] += lazy[pos];
                }
                lazy[pos] = 0;
            }
            if (s >= gs && e <= ge) {
                return tree[pos];
            }
            int m = (s + e) / 2;
            return Math.max(getUtil(gs, ge, s, m, 2 * pos + 1), getUtil(gs, ge, m + 1, e, 2 * pos + 2));
        }

        private void updateUtil(int us, int ue, long val, int s, int e, int pos) {
            if (s > e || s > ue || e < us) return;

            if (lazy[pos] != 0) {
                tree[pos] += lazy[pos];
                if (s != e) {
                    lazy[2 * pos + 1] += lazy[pos];
                    lazy[2 * pos + 2] += lazy[pos];
                }
                lazy[pos] = 0;
            }

            if (s >= us && e <= ue) {
                tree[pos] += val;
                if (s != e) {
                    lazy[2 * pos + 1] += val;
                    lazy[2 * pos + 2] += val;
                }
                return;
            }

            int m = (s + e) / 2;
            updateUtil(us, ue, val, s, m, 2 * pos + 1);
            updateUtil(us, ue, val, m + 1, e, 2 * pos + 2);

            tree[pos] = Math.max(tree[2 * pos + 1], tree[2 * pos + 2]);
        }


    }


    static int[] h = {0, 0, -1, 1};
    static int[] v = {1, -1, 0, 0};


    public static class Pair {
        public int x, y, p;

        public Pair(int d, int y) {
            this.x = d;
            this.y = y;
        }


    }

    static long compute_hash(String s) {
        int p = 31;
        int m = 1000000007;
        long hash_value = 0;
        long p_pow = 1;
        for (int i = 0; i < s.length(); ++i) {
            char c = s.charAt(i);
            hash_value = (hash_value + (c - 'a' + 1) * p_pow) % m;
            p_pow = (p_pow * p) % m;
        }
        return hash_value;
    }

    static int counter = 0;

    public static void main(String[] args) throws Exception {
        //https://i...content-available-to-author-only...e.com/ebRGa6
        InputReader in = new InputReader(System.in);
        long t = in.readLong();

            int[][] pre = new int[100000+1][17];
            int[] tin = new int[100000+1];
            int[] tout = new int[100000+1];
            int[] xors = new int[100000+1];
        while (t-- > 0) {
            counter = 0;
            int n = in.readInt();
            int q = in.readInt();

            int[] val = new int[n+1];
            for (int i = 1; i <= n; ++i) {
                val[i] = in.readInt();
            }

            List<List<Integer>> tree = new ArrayList<>();
            for (int i = 0; i <= n; ++i) {
                tree.add(new ArrayList<>());
                Arrays.fill(pre[i], -1);
            }
            for (int i = 0; i < n - 1; ++i) {
                int p = in.readInt();
                int r = in.readInt();

                tree.get(p).add(r);
                tree.get(r ).add(p );

            }

            preprocess(1, pre, tree, new int[n+1], 0, -1, tin, tout, 0, val, xors);
//            for (int i = 0; i < n; ++i) {
//                for (int j = 30; j >= 0; --j) {
//                    System.out.print(pre[i][j] + " ");
//                }
//                System.out.println("");
//            }
            while (q-- > 0) {
                int s = in.readInt();
                int e = in.readInt();
                if (e > s) {
                    int temp = s;
                    s = e;
                    e = temp;
                }
               
                if (s == e) {
                    System.out.println(val[s]);
                    continue;
                }
                
                if ((tin[s] <= tin[e] && tout[s] >= tout[e])) {
                    
                    System.out.println(xors[s] ^ xors[e] ^ val[s]);
                    continue;
                }
                if ((tin[s] >= tin[e] && tout[s] <= tout[e])) {
                    
                    System.out.println(xors[s] ^ xors[e] ^ val[e]);
                    continue;
                }
                int xor = xors[s];
                for (int i = 16; i >= 0; --i) {
                    if (pre[s][i] != 0) {
                        if (!(tin[pre[s][i]] <= tin[e] && tout[pre[s][i]] >= tout[e])) {
                            s = pre[s][i];
                        }
                    }
                }
                // assert(pre[s][0] != -1);
                xor = xor ^ xors[e] ^ val[pre[s][0]];
                System.out.println(xor);
            }

        }


    }

    private static void preprocess(int pos, int[][] pre, List<List<Integer>> tree, int[] traverse, int depth, int last, int[] tin, int[] tout, int xor, int[] val, int[] xors) {
        tin[pos] = counter++;
        xors[pos] = xor ^ val[pos];
        traverse[depth] = pos;

        for (int i = 0; depth - (1 << i) >= 0; ++i) {
            pre[pos][i] = traverse[depth - (1 << i)];
        }

        for (int i = 0; i < tree.get(pos).size(); ++i) {
            if (tree.get(pos).get(i) != last)
                preprocess(tree.get(pos).get(i), pre, tree, traverse, depth + 1, pos, tin, tout, xors[pos], val, xors);
        }
        tout[pos] = counter++;
    }

    private static long solve(long[][] dp, int n, int r) {
        if (n == 0) {
            if (r == 0) return 1;
            else return 0;
        }
        if (r <= 0) return 0;
        if (dp[n][r] != -1) return dp[n][r];
        long answer = 0;
        for (int i = 1; i < n; ++i) {
            answer += solve(dp, n - 1, r - i);
        }
        dp[n][r] = answer;
        return answer;
    }

    static int gcd(int a, int b) {

        while (b != 0) {
            int t = a;
            a = b;
            b = t % b;
        }
        return a;
    }

    class Solution {
        public int maxProduct(int[] nums) {
            PriorityQueue<Integer> p = new PriorityQueue<>(new Comparator<Integer>() {
                @Override
                public int compare(Integer integer, Integer t1) {
                    return Integer.compare(t1, integer);
                }
            });
            for (int i = 0; i < nums.length; ++i) {
                p.add(nums[i]);
            }
            int f = p.remove();
            int s = p.remove();
            return (f - 1) * (s - 1);
        }
    }

    static boolean submit = true;

    static void debug(String s) {
        if (!submit)
            System.out.println(s);
    }

    static void debug(int s) {
        if (!submit)
            System.out.println(s);
    }

}

































