package main

import (
	"bufio"
	"fmt"
	"os"
	"sort"
	"strconv"
)

func Assert(ok bool) {
	if !ok {
		panic("assertion failed")
	}
}

var sc = func() *bufio.Scanner {
	sc := bufio.NewScanner(os.Stdin)
	sc.Split(bufio.ScanWords)
	return sc
}()

func Str() string {
	sc.Scan()
	return sc.Text()
}

func Int() int {
	x, _ := strconv.Atoi(Str())
	return x
}

func Float() float64 {
	x, _ := strconv.ParseFloat(Str(), 64)
	return x
}

var wr = bufio.NewWriter(os.Stdout)

func Out(a ...interface{}) {
	fmt.Fprintln(wr, a...)
}

func Format(a []int) string {
	s := fmt.Sprint(a)
	return s[1 : len(s)-1]
}

func Chmin(a *int, b int) bool {
	if b < *a {
		*a = b
		return true
	}
	return false
}

func Chmax(a *int, b int) bool {
	if b > *a {
		*a = b
		return true
	}
	return false
}

func Max(a ...int) int {
	mx := a[0]
	for _, x := range a {
		if x > mx {
			mx = x
		}
	}
	return mx
}

func Min(a ...int) int {
	mn := a[0]
	for _, x := range a {
		if x < mn {
			mn = x
		}
	}
	return mn
}

func Sum(a ...int) int {
	s := 0
	for _, x := range a {
		s += x
	}
	return s
}

// truncated
func Divmod(a, b int) (int, int) {
	q := a / b
	r := a - q*b
	return q, r
}

func Lb(a []int, x int) int {
	n := len(a)
	f := func(i int) bool {
		return a[i] >= x
	}
	return sort.Search(n, f)
}

func Ub(a []int, x int) int {
	n := len(a)
	f := func(i int) bool {
		return a[i] > x
	}
	return sort.Search(n, f)
}

func Iota(n, v int) []int {
	a := make([]int, n)
	for i := 0; i < n; i++ {
		a[i] = v + i
	}
	return a
}

func YesNo(
	y, n string,
) func(bool) {
	return func(f bool) {
		if f {
			Out(y)
		} else {
			Out(n)
		}
	}
}

var Yes = YesNo("Yes", "No")

// var Yes = YesNo("yes", "no")
// var Yes = YesNo("YES", "NO")

func main() {
	defer wr.Flush()
	t := 1
	// t := Int()
	for ; t > 0; t-- {
		Solve()
	}
}

func Egcd(a, b int) (int, int, int) {
	if b == 0 {
		if a < 0 {
			return -a, -1, 0
		} else {
			return a, 1, 0
		}
	}
	q, r := Divmod(a, b)
	g, s, t := Egcd(b, r)
	return g, t, s - q*t
}

func Inv(m, a int) int {
	Assert(0 < a && a < m)
	g, x, _ := Egcd(a, m)
	Assert(g == 1)
	if x < 0 {
		x += m
	}
	return x
}

const mod = 998244353

type Comb struct {
	m          int
	F, Fi, Inv []int
}

// new(Comb).Init()
func (c *Comb) Init(m, n int) *Comb {
	c.m = m
	f := make([]int, n)
	fi := make([]int, n)
	iv := make([]int, n)
	f[0] = 1
	for i := 1; i < n; i++ {
		f[i] = f[i-1] * i % m
	}
	fi[n-1] = Inv(m, f[n-1])
	for i := n - 1; i >= 1; i-- {
		fi[i-1] = fi[i] * i % m
		iv[i] = fi[i] * f[i] % m
	}
	c.F, c.Fi, c.Inv = f, fi, iv
	return c
}

func (c *Comb) P(n, k int) int {
	if k < 0 || n < k {
		return 0
	}
	return c.F[n] * c.Fi[n-k] % c.m
}

func (c *Comb) C(n, k int) int {
	if k < 0 || n < k {
		return 0
	}
	return c.P(n, k) * c.Fi[k] % c.m
}

func (c *Comb) H(n, k int) int {
	return c.C(n-1+k, k)
}

func (c *Comb) IP(n, k int) int {
	Assert(0 <= k && k <= n)
	return c.Fi[n] * c.F[n-k] % c.m
}

func (c *Comb) IC(n, k int) int {
	return c.IP(n, k) * c.F[k] % c.m
}

func Solve() {
	x := Int()
	y := Int()
	z := Int()
	w := Int()
	Chmax(&z, 1)
	Chmax(&w, 1)
	f := new(Comb).Init(mod, 1<<20)
	res := f.F[x+y-z-w]
	res *= f.C(x, z)
	res %= mod
	res *= f.C(y, w)
	res %= mod
	Out(res)
}