validation_problems.py 3.21 KB
Newer Older
mashun1's avatar
jax-cfd  
mashun1 committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
# Copyright 2021 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#      http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

"""Descriptions and analytical solutions for validation problems."""

import abc
from typing import Optional, Sequence, Tuple

import jax.numpy as jnp
from jax_cfd.base import boundaries
from jax_cfd.base import grids
import numpy as np

GridArray = grids.GridArray
GridArrayVector = grids.GridArrayVector
GridVariableVector = grids.GridVariableVector
Offsets = Sequence[Sequence[float]]


class Problem(metaclass=abc.ABCMeta):
  """An abstract class for Navier-Stokes problems."""

  @property
  def grid(self):
    return self._grid  # pytype: disable=attribute-error  # bind-properties

  @property
  def density(self):
    return self._density  # pytype: disable=attribute-error  # bind-properties

  @property
  def viscosity(self):
    return self._viscosity  # pytype: disable=attribute-error  # bind-properties

  def force(self,
            offsets: Optional[Offsets] = None) -> Optional[GridArrayVector]:
    del offsets  # Unused
    return None

  @abc.abstractmethod
  def velocity(self,
               t: float,
               offsets: Optional[Offsets] = None) -> GridVariableVector:
    pass


class TaylorGreen(Problem):
  """2D Taylor Green vortices with analytic solution for velocity.

  See https://en.wikipedia.org/wiki/Taylor%E2%80%93Green_vortex.
  """
  # TODO(jamieas): consider parameterizing problems in terms of Reynolds
  # number.

  def __init__(self,
               shape: Tuple[int, int],
               density: float = 1,
               viscosity: float = 0,
               kx: float = 1,
               ky: float = 1):
    self._grid = grids.Grid(shape=shape,
                            domain=[(0., 2. * np.pi),
                                    (0., 2. * np.pi)])
    self._density = density
    self._viscosity = viscosity
    self._kx = kx
    self._ky = ky

  def velocity(
      self,
      t: float = 0,
      offsets: Optional[Offsets] = None) -> GridVariableVector:
    """Returns an analytic solution for velocity at time `t`."""
    if offsets is None:
      offsets = self.grid.cell_faces

    scale = jnp.exp(-2 * self.viscosity * t)

    ux, uy = self.grid.mesh(offsets[0])
    u = grids.GridVariable(
        array=grids.GridArray(
            data=scale * jnp.cos(self._kx * ux) * jnp.sin(self._ky * uy),
            offset=offsets[0],
            grid=self.grid),
        bc=boundaries.periodic_boundary_conditions(self.grid.ndim))

    vx, vy = self.grid.mesh(offsets[1])
    v = grids.GridVariable(
        array=grids.GridArray(
            data=-scale * jnp.sin(self._kx * vx) * jnp.cos(self._ky * vy),
            offset=offsets[1],
            grid=self.grid),
        bc=boundaries.periodic_boundary_conditions(self.grid.ndim))

    return (u, v)