#include #include #include #include

using namespace std;

typedef pair <double, int> dpr;

const int maxn = 30009; const int maxl = 15; const int max_buf = maxn * maxl * 2; const double eps = 1e-7; const double finf = 1e20;

int ibuf_arr[max_buf], *ibuf; double fbuf_arr[max_buf], *fbuf;

struct edge { int t; edge *next; };

struct line { double k, b; inline double xCross(const line& a) { return (a. b - b) / (k - a. k); } inline double f(double x) { return k * x + b; } }; line a[maxn « 1];

struct hull { int t; int *l; double *cx; void init(int x) { t = 1; l = ibuf ++; l[0] = x; cx = 0; } void merge(const hull& x, const hull& y) { static int stc[maxn]; static double tx[maxn]; t = 0; for (int i = 0, j = 0, c; i < x. t || j < y. t; ) { if (i == x. t || (j < y. t && a[x. l[i]]. k > a[y. l[j]]. k)) c = y. l[j ++]; else c = x. l[i ++]; if (t && fabs(a[c]. k - a[stc[t - 1]]. k) < eps) { if (a[c]. b >= a[stc[t - 1]]. b) – t; else continue; } while (t > 1 && a[c]. xCross(a[stc[t - 2]]) <= tx[t - 2]) – t; if (t) tx[t - 1] = a[c]. xCross(a[stc[t - 1]]); stc[t ++] = c; } l = ibuf; ibuf += t; cx = fbuf; fbuf += t - 1; for (int i = 0; i < t; ++ i) { l[i] = stc[i]; if (i < t - 1) cx[i] = a[stc[i]]. xCross(a[stc[i + 1]]); } } dpr qry(double x) { int p(lower_bound(cx, cx + t - 1, x) - cx); return dpr(a[l[p]]. f(x), l[p]); } }; struct seg { hull h[2]; int l, r; seg *ls, *rs; };

int n, m, k1[maxn], b1[maxn], k2[maxn], b2[maxn]; dpr qans[2]; double qx; int fa[maxn], d[maxn], sz[maxn], tc, fc[maxn], ch[maxn], cl[maxn], *ci[maxn]; edge elst[maxn « 1], *ep, *head[maxn]; seg slst[maxn « 2], *sp, *rt[maxn];

inline void upMax(double& a, double b) { if (a < b) a = b; } inline void upMax(dpr& a, dpr b) { if (a < b) a = b; } inline void addEdge(int u, int v) { ep-> t = v; ep-> next = head[u]; head[u] = ep ++; }

#define cpos(p) (d[p]-d[ch[fc[p]]]) #define midp(p) ((p->l+p->r)»1)

inline seg* sgtBuild(int l, int r, int* ci) { seg p(sp ++); p-> l = l; p-> r = r; if (l + 1 == r) { p-> h[0]. init(ci[l]); p-> h[1]. init(ci[l] + n); } else { p-> ls = sgtBuild(l, midp(p), ci); p-> rs = sgtBuild(midp(p), r, ci); p-> h[0]. merge(p-> ls-> h[0], p-> rs-> h[0]); p-> h[1]. merge(p-> ls-> h[1], p-> rs-> h[1]); } return p; } inline void sgtQry(seg p, int l, int r) { if (p-> l == l && p-> r == r) { upMax(qans[0], p-> h[0]. qry(qx)); upMax(qans[1], p-> h[1]. qry(qx)); } else if (r <= midp(p)) sgtQry(p-> ls, l, r); else if (l >= midp(p)) sgtQry(p-> rs, l, r); else { sgtQry(p-> ls, l, midp(p)); sgtQry(p-> rs, midp(p), r); } }

void divide() { static int q[maxn]; int hd(0), tl(1); memset(d, 0, sizeof(d)); d[q[0] = 1] = 1; fa[1] = 0; while (hd < tl) { int p(q[hd ++]); for (edge* e = head[p]; e; e = e-> next) if (!d[e-> t]) { d[e-> t] = d[p] + 1; fa[e-> t] = p; q[tl ++] = e-> t; } } tc = 0; for (int i = tl - 1; i >= 0; – i) { int p(q[i]), z(-1); sz[p] = 1; for (edge* e = head[p]; e; e = e-> next) if (fa[e-> t] == p) { sz[p] += sz[e-> t]; if (z == -1 || sz[e-> t] > sz[z]) z = e-> t; } if (z == -1) cl[fc[p] = ++ tc] = 1; else ++ cl[fc[p] = fc[z]]; ch[fc[p]] = p; } for (int i = 1; i <= tc; ++ i) { ci[i] = ibuf; ibuf += cl[i]; } for (int i = 1; i <= n; ++ i) ci[fc[i]][cpos(i)] = i; for (int i = 1; i <= tc; ++ i) rt[i] = sgtBuild(0, cl[i], ci[i]); }

bool check(int u, int v, double x) { qans[0] = qans[1] = dpr(-finf, 0); qx = -x; while (fc[u] ^ fc[v]) if (d[ch[fc[u]]] > d[ch[fc[v]]]) { sgtQry(rt[fc[u]], 0, cpos(u) + 1); u = fa[ch[fc[u]]]; } else { sgtQry(rt[fc[v]], 0, cpos(v) + 1); v = fa[ch[fc[v]]]; } if (d[u] < d[v]) sgtQry(rt[fc[u]], cpos(u), cpos(v) + 1); else sgtQry(rt[fc[u]], cpos(v), cpos(u) + 1); return qans[0]. first + qans[1]. first >= 0; }

int gcd(int a, int b) { while (a) { int c(b % a); b = a; a = c; } return b; }

int main() { while (scanf("%d%d", &n, &m) != EOF) { for (int i = 1; i <= n; ++ i) { scanf("%d%d%d%d", k1 + i, b1 + i, k2 + i, b2 + i); a[i]. k = k1[i]; a[i]. b = a[i]. k * b1[i]; a[i + n]. k = k2[i]; a[i + n]. b = a[i + n]. k * b2[i]; } memset(head, 0, sizeof(head)); ibuf = ibuf_arr; fbuf = fbuf_arr; ep = elst; sp = slst; for (int i = 1; i < n; ++ i) { int u, v; scanf("%d%d", &u, &v); addEdge(u, v); addEdge(v, u); } divide(); while (m –) { int u, v; scanf("%d%d", &u, &v); double l(1e-3), r(1e3); while (l + eps < r) { double mid((l + r + eps) / 2.0); if (check(u, v, mid)) l = mid; else r = mid - eps; } int x(qans[0]. second), y(qans[1]. second - n); int fu((k1[x] * b1[x] + k2[y] * b2[y])); int fd(k1[x] + k2[y]); int g(gcd(fu, fd)); fu /= g, fd /= g; printf("%d/%d\n", fu, fd); } } }