{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "\n", "# Start-to-Finish Example: Numerical Solution of the Scalar Wave Equation, in Cartesian Coordinates\n", "\n", "## Author: Zach Etienne\n", "### Formatting improvements courtesy Brandon Clark\n", "\n", "## This module solves the scalar wave equation in Cartesian coordinates, using the [Method of Lines](Tutorial-Method_of_Lines-C_Code_Generation.ipynb).\n", "\n", "**Notebook Status:** Validated\n", "\n", "**Validation Notes:** This module has been validated to converge at the expected order to the exact solution (see [plot](#convergence) at bottom).\n", "\n", "### NRPy+ Source Code for this module: \n", "* [ScalarWave/ScalarWave_RHSs.py](../edit/ScalarWave/ScalarWave_RHSs.py) [\\[**tutorial**\\]](Tutorial-ScalarWave.ipynb) Generates the right-hand side for the Scalar Wave Equation in cartesian coordinates\n", "* [ScalarWave/InitialData.py](../edit/ScalarWave/InitialData.py) [\\[**tutorial**\\]](Tutorial-ScalarWave.ipynb) Generates C code for plane wave or spherical Gaussian initial data for the scalar wave equation\n", "\n", "## Introduction:\n", "\n", "As outlined in the [previous NRPy+ tutorial notebook](Tutorial-ScalarWave.ipynb), we first use NRPy+ to generate initial data for the scalar wave equation, and then we use it to generate the RHS expressions for [Method of Lines](https://reference.wolfram.com/language/tutorial/NDSolveMethodOfLines.html) time integration based on the [explicit Runge-Kutta fourth-order scheme](https://en.wikipedia.org/wiki/Runge%E2%80%93Kutta_methods) (RK4).\n", "\n", "The entire algorithm is outlined as follows, with links to the relevant NRPy+ tutorial notebooks listed at each step:\n", "\n", "1. Allocate memory for gridfunctions, including temporary storage for the Method of Lines time integration\n", " * [**NRPy+ tutorial notebook on Method of Lines algorithm**](Tutorial-Method_of_Lines-C_Code_Generation.ipynb).\n", "1. Set gridfunction values to initial data \n", " * [**NRPy+ tutorial notebook section on plane-wave solution to scalar wave equation**](Tutorial-ScalarWave.ipynb#planewavesoln)\n", "1. Next, integrate the initial data forward in time using the Method of Lines coupled to a Runge-Kutta explicit timestepping algorithm:\n", " 1. At the start of each iteration in time, output the difference between the numerical and exact solution\n", " * [**NRPy+ tutorial notebook section on plane-wave solution to scalar wave equation**](Tutorial-ScalarWave.ipynb#planewavesoln).\n", " 1. At each RK time substep, do the following:\n", " 1. Evaluate scalar wave RHS expressions \n", " * [**NRPy+ tutorial notebook section on right-hand sides of scalar wave equation, in 3 spatial dimensions**](Tutorial-ScalarWave.ipynb#rhss3d)\n", " 1. Apply boundary conditions [*a la* the SENR/NRPy+ paper](https://arxiv.org/abs/1712.07658)\n", "1. Repeat above steps at two numerical resolutions to confirm convergence to zero." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "# Table of Contents\n", "$$\\label{toc}$$\n", "\n", "This notebook is organized as follows\n", "\n", "1. [Step 1](#setup): Set up core functions and parameters for solving scalar wave equation\n", " 1. [Step 1.a](#applybcs) `apply_bcs()`: outer boundary condition driver function\n", " 1. [Step 1.b](#mol) Generate Runge-Kutta-based Method of Lines timestepping code\n", " 1. [Step 1.c](#freeparams) Output C codes needed for declaring and setting Cparameters; also set `free_parameters.h`\n", "1. [Step 2](#mainc): `ScalarWave_Playground.c`: The Main C Code\n", "1. [Step 3](#convergence): Code validation: Verify that relative error in numerical solution converges to zero at the expected order\n", "1. [Step 4](#latex_pdf_output): Output this notebook to $\\LaTeX$-formatted PDF file" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "# Step 1: Set up core functions and parameters for solving scalar wave equation \\[Back to [top](#toc)\\]\n", "$$\\label{setup}$$\n", "\n", "Let's pick up where we left off in the [previous module](Tutorial-ScalarWave.ipynb), interfacing with the [ScalarWave/InitialData](../edit/ScalarWave/InitialData.py) and [ScalarWave/ScalarWave_RHSs](../edit/ScalarWave/ScalarWave_RHSs.py) NRPy+ modules to generate\n", "* monochromatic (single-wavelength) plane wave scalar wave initial data, and\n", "* the scalar wave equation RHSs at **4th** finite difference order in **3 spatial dimensions**" ] }, { "cell_type": "code", "execution_count": 1, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:43.423384Z", "iopub.status.busy": "2021-03-07T17:32:43.416415Z", "iopub.status.idle": "2021-03-07T17:32:44.006636Z", "shell.execute_reply": "2021-03-07T17:32:44.007147Z" } }, "outputs": [], "source": [ "# Step P1: Import needed NRPy+ core modules:\n", "from outputC import lhrh,outCfunction # NRPy+: Core C code output module\n", "import finite_difference as fin # NRPy+: Finite difference C code generation module\n", "import NRPy_param_funcs as par # NRPy+: Parameter interface\n", "import grid as gri # NRPy+: Functions having to do with numerical grids\n", "import indexedexp as ixp # NRPy+: Symbolic indexed expression (e.g., tensors, vectors, etc.) support\n", "import cmdline_helper as cmd # NRPy+: Multi-platform Python command-line interface\n", "import shutil, os, sys # Standard Python modules for multiplatform OS-level functions\n", "\n", "# Step P2: Create C code output directory:\n", "Ccodesdir = os.path.join(\"ScalarWave_Ccodes/\")\n", "# First remove C code output directory if it exists\n", "# Courtesy https://stackoverflow.com/questions/303200/how-do-i-remove-delete-a-folder-that-is-not-empty\n", "# !rm -r ScalarWaveCurvilinear_Playground_Ccodes\n", "shutil.rmtree(Ccodesdir, ignore_errors=True)\n", "# Then create a fresh directory\n", "cmd.mkdir(Ccodesdir)\n", "\n", "# Step P3: Create executable output directory:\n", "outdir = os.path.join(Ccodesdir,\"output/\")\n", "cmd.mkdir(outdir)\n", "\n", "# Step P4: Set domain_size, the physical extent of numerical grid;\n", "# in Cartesian coordinates xmin=ymin=zmin=-domain_size,\n", "# and xmax=ymax=zmax=+domain_size\n", "domain_size = 10.0\n", "\n", "# Step P5: Set timestepping algorithm (we adopt the Method of Lines)\n", "RK_method = \"RK4\"\n", "\n", "# Step P6: Set the finite differencing order to 4.\n", "par.set_parval_from_str(\"finite_difference::FD_CENTDERIVS_ORDER\",4)\n", "\n", "# Step 1: Import the ScalarWave.InitialData module.\n", "# This command only declares ScalarWave initial data\n", "# parameters and the InitialData() function.\n", "import ScalarWave.InitialData as swid\n", "\n", "# Step 2: Import ScalarWave_RHSs module.\n", "# This command only declares ScalarWave RHS parameters\n", "# and the ScalarWave_RHSs function (called later)\n", "import ScalarWave.ScalarWave_RHSs as swrhs\n", "\n", "# Step 3: Set the spatial dimension parameter\n", "# to 3, and then read the parameter as DIM.\n", "par.set_parval_from_str(\"grid::DIM\",3)\n", "DIM = par.parval_from_str(\"grid::DIM\")\n", "\n", "# Step 4: Call the InitialData() function to set up initial data.\n", "# Options include:\n", "# \"PlaneWave\": monochromatic (single frequency/wavelength) plane wave\n", "# \"SphericalGaussian\": spherically symmetric Gaussian, with default stdev=3\n", "swid.InitialData(Type=\"PlaneWave\")\n", "\n", "# Step 5: Generate SymPy symbolic expressions for\n", "# uu_rhs and vv_rhs; the ScalarWave RHSs.\n", "# This function also declares the uu and vv\n", "# gridfunctions, which need to be declared\n", "# to output even the initial data to C file.\n", "swrhs.ScalarWave_RHSs()\n", "\n", "# Step 6: Enable \"FD functions\". In other words, all finite-difference stencils\n", "# will be output as inlined static functions. This is essential for\n", "# compiling highly complex FD kernels with using certain versions of GCC;\n", "# GCC 10-ish will choke on BSSN FD kernels at high FD order, sometimes\n", "# taking *hours* to compile. Unaffected GCC versions compile these kernels\n", "# in seconds. FD functions do not slow the code performance, but do add\n", "# another header file to the C source tree.\n", "enable_FD_functions = True\n", "par.set_parval_from_str(\"finite_difference::enable_FD_functions\", enable_FD_functions)" ] }, { "cell_type": "code", "execution_count": 2, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:43.423384Z", "iopub.status.busy": "2021-03-07T17:32:43.416415Z", "iopub.status.idle": "2021-03-07T17:32:44.006636Z", "shell.execute_reply": "2021-03-07T17:32:44.007147Z" }, "scrolled": false }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Output C function exact_solution_single_point() to file ScalarWave_Ccodes/exact_solution_single_point.h\n", "Output C function exact_solution_all_points() to file ScalarWave_Ccodes/exact_solution_all_points.h\n", "Output C function rhs_eval() to file ScalarWave_Ccodes/rhs_eval.h\n" ] } ], "source": [ "desc=\"Part P3: Declare the function for the exact solution at a single point. time==0 corresponds to the initial data.\"\n", "name=\"exact_solution_single_point\"\n", "outCfunction(\n", " outfile = os.path.join(Ccodesdir,name+\".h\"), desc=desc, name=name,\n", " params =\"const REAL xx0,const REAL xx1,const REAL xx2,const paramstruct *restrict params,REAL *uu_exact,REAL *vv_exact\",\n", " body = fin.FD_outputC(\"returnstring\",[lhrh(lhs=\"*uu_exact\",rhs=swid.uu_ID),\n", " lhrh(lhs=\"*vv_exact\",rhs=swid.vv_ID)]),\n", " loopopts = \"\")\n", "\n", "\n", "desc=\"Part P4: Declare the function for the exact solution at all points. time==0 corresponds to the initial data.\"\n", "name=\"exact_solution_all_points\"\n", "outCfunction(\n", " outfile = os.path.join(Ccodesdir,name+\".h\"), desc=desc, name=name,\n", " params =\"const paramstruct *restrict params,REAL *restrict xx[3], REAL *restrict in_gfs\",\n", " body =\"exact_solution_single_point(xx0,xx1,xx2,params,&in_gfs[IDX4S(UUGF,i0,i1,i2)],&in_gfs[IDX4S(VVGF,i0,i1,i2)]);\",\n", " loopopts = \"AllPoints,Read_xxs\")\n", "\n", "\n", "desc=\"Part P5: Declare the function to evaluate the scalar wave RHSs\"\n", "includes = None\n", "if enable_FD_functions:\n", " includes = [\"finite_difference_functions.h\"]\n", "name=\"rhs_eval\"\n", "outCfunction(\n", " outfile = os.path.join(Ccodesdir,name+\".h\"), includes=includes, desc=desc, name=name,\n", " params =\"const paramstruct *restrict params, const REAL *restrict in_gfs, REAL *restrict rhs_gfs\",\n", " body =fin.FD_outputC(\"returnstring\",[lhrh(lhs=gri.gfaccess(\"rhs_gfs\",\"uu\"),rhs=swrhs.uu_rhs),\n", " lhrh(lhs=gri.gfaccess(\"rhs_gfs\",\"vv\"),rhs=swrhs.vv_rhs)],\n", " params=\"enable_SIMD=True\"),\n", " loopopts = \"InteriorPoints,enable_SIMD\")\n", "\n", "# Step 6.b Output functions for computing all finite-difference stencils\n", "if enable_FD_functions:\n", " fin.output_finite_difference_functions_h(path=Ccodesdir)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "## Step 1.a: `apply_bcs()`: outer boundary condition driver function\n", "$$\\label{applybcs}$$\n", "\n", "When solving the wave equation on a 3D Cartesian numerical grid cube (or, if you like, rectangular prism), at each step in time, we first evaluate the right-hand sides (RHSs) of the $\\partial_t u$ and $\\partial_t v$ equations. \n", "\n", "These RHSs generally contain spatial derivatives, which we evaluate using finite-difference differentiation ([**tutorial**](Tutorial-Finite_Difference_Derivatives.ipynb)). Each finite-difference derivative depends on neighboring points on the left and right, so the RHSs can only be evaluated in the grid interior. For example, a standard fourth-order centered finite difference derivative depends on two points to the left and right of the point at which the derivative is being evaluated. In order for the same interior to be filled at the next time step, we need to fill in the data at the boundaries; i.e., we need to apply boundary conditions.\n", "\n", "Here we quadratically extrapolate data to the outer boundary using the `FACE_UPDATE()` C macro defined below. The C code function `apply_bcs()` below updates all 6 faces of the cube. To ensure that all gridpoints on the outer boundary (also known as \"ghost cells\") are filled, the following algorithm is implemented, starting at the innermost ghost cells (i.e., the ghost cells closest to the grid interior):\n", "\n", "1. The lower $x$ face is updated on only the interior points of the face.\n", "1. The upper $x$ face is updated on only the interior points of the face.\n", "1. The lower $y$ face is updated on the interior points of that face, plus the lower and upper $x$ boundary points\n", "1. The upper $y$ face is updated on the interior points of that face, plus the lower and upper $x$ boundary points\n", "1. The lower $z$ face is updated on the interior points of that face, plus the lower and upper $x$ boundary points, plus the lower and upper $y$ boundary points\n", "1. The upper $z$ face is updated on the interior points of that face, plus the lower and upper $x$ boundary points, plus the lower and upper $y$ boundary points\n", "1. The above is repeated on the next outer ghost cell, until all outer boundary points are filled." ] }, { "cell_type": "code", "execution_count": 3, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:44.013381Z", "iopub.status.busy": "2021-03-07T17:32:44.012289Z", "iopub.status.idle": "2021-03-07T17:32:44.016350Z", "shell.execute_reply": "2021-03-07T17:32:44.016939Z" } }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Writing ScalarWave_Ccodes//apply_bcs.h\n" ] } ], "source": [ "%%writefile $Ccodesdir/apply_bcs.h\n", "\n", "// Declare boundary condition FACE_UPDATE macro,\n", "// which updates a single face of the 3D grid cube\n", "// using quadratic polynomial extrapolation.\n", "const int MAXFACE = -1;\n", "const int NUL = +0;\n", "const int MINFACE = +1;\n", "#define FACE_UPDATE(which_gf, i0min,i0max, i1min,i1max, i2min,i2max, FACEX0,FACEX1,FACEX2) \\\n", " for(int i2=i2min;i2\n", "\n", "## Step 1.b: Generate Runge-Kutta-based Method of Lines timestepping code \\[Back to [top](#toc)\\]\n", "$$\\label{mol}$$\n", "\n", "The Method of Lines algorithm is described in detail in the [**NRPy+ tutorial notebook on Method of Lines algorithm**](Tutorial-Method_of_Lines-C_Code_Generation.ipynb)." ] }, { "cell_type": "code", "execution_count": 4, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:44.023378Z", "iopub.status.busy": "2021-03-07T17:32:44.022377Z", "iopub.status.idle": "2021-03-07T17:32:44.037707Z", "shell.execute_reply": "2021-03-07T17:32:44.036956Z" } }, "outputs": [], "source": [ "# Step 1.b: Generate Runge-Kutta-based (RK-based) timestepping code.\n", "# As described above the Table of Contents, this is a 2-step process:\n", "# 1.b.A: Evaluate RHSs (RHS_string)\n", "# 1.b.B: Apply boundary conditions (post_RHS_string, pt 1)\n", "import MoLtimestepping.C_Code_Generation as MoL\n", "from MoLtimestepping.RK_Butcher_Table_Dictionary import Butcher_dict\n", "RK_order = Butcher_dict[RK_method][1]\n", "cmd.mkdir(os.path.join(Ccodesdir,\"MoLtimestepping/\"))\n", "MoL.MoL_C_Code_Generation(RK_method,\n", " RHS_string = \"rhs_eval(¶ms, RK_INPUT_GFS, RK_OUTPUT_GFS);\",\n", " post_RHS_string = \"apply_bcs(¶ms, RK_OUTPUT_GFS);\",\n", " outdir = os.path.join(Ccodesdir,\"MoLtimestepping/\"))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "## Step 1.c: Output C codes needed for declaring and setting Cparameters; also set `free_parameters.h` \\[Back to [top](#toc)\\]\n", "$$\\label{freeparams}$$\n", "\n", "Based on declared NRPy+ Cparameters, first we generate `declare_Cparameters_struct.h`, `set_Cparameters_default.h`, and `set_Cparameters[-SIMD].h`.\n", "\n", "Then we output `free_parameters.h`, which sets initial data parameters, as well as grid domain & reference metric parameters, applying `domain_size` and `sinh_width`/`SymTP_bScale` (if applicable) as set above" ] }, { "cell_type": "code", "execution_count": 5, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:44.044871Z", "iopub.status.busy": "2021-03-07T17:32:44.044110Z", "iopub.status.idle": "2021-03-07T17:32:44.048339Z", "shell.execute_reply": "2021-03-07T17:32:44.048859Z" } }, "outputs": [], "source": [ "# Step 3.d.i: Generate declare_Cparameters_struct.h, set_Cparameters_default.h, and set_Cparameters[-SIMD].h\n", "par.generate_Cparameters_Ccodes(os.path.join(Ccodesdir))\n", "\n", "domain_size_str=str(domain_size)\n", "# Step 3.d.ii: Set free_parameters.h\n", "with open(os.path.join(Ccodesdir,\"free_parameters.h\"),\"w\") as file:\n", " file.write(\"\"\"\n", "// Set free-parameter values.\n", "\n", "// Set free-parameter values for the initial data.\n", "params.time = 0.0; params.wavespeed = 1.0;\n", "//params.kk0 = 1.0; params.kk1 = 1.0; params.kk2 = 1.0;\n", "\n", "const REAL domain_size = \"\"\"+str(domain_size)+\"\"\";\n", "\n", "// Override parameter defaults with values based on command line arguments and NGHOSTS.\n", "const int Nx0x1x2 = atoi(argv[1]);\n", "params.Nxx0 = Nx0x1x2;\n", "params.Nxx1 = Nx0x1x2;\n", "params.Nxx2 = Nx0x1x2;\n", "params.Nxx_plus_2NGHOSTS0 = params.Nxx0 + 2*NGHOSTS;\n", "params.Nxx_plus_2NGHOSTS1 = params.Nxx1 + 2*NGHOSTS;\n", "params.Nxx_plus_2NGHOSTS2 = params.Nxx2 + 2*NGHOSTS;\n", "// Step 0d: Set up space and time coordinates\n", "// Step 0d.i: Declare \\Delta x^i=dxx{0,1,2} and invdxx{0,1,2}, as well as xxmin[3] and xxmax[3]:\n", "const REAL xxmin[3] = {-\"\"\"+domain_size_str+\"\"\",-\"\"\"+domain_size_str+\"\"\",-\"\"\"+domain_size_str+\"\"\" };\n", "const REAL xxmax[3] = {+\"\"\"+domain_size_str+\"\"\",+\"\"\"+domain_size_str+\"\"\",+\"\"\"+domain_size_str+\"\"\" };\n", "\n", "params.dxx0 = (xxmax[0] - xxmin[0]) / ((REAL)params.Nxx0);\n", "params.dxx1 = (xxmax[1] - xxmin[1]) / ((REAL)params.Nxx1);\n", "params.dxx2 = (xxmax[2] - xxmin[2]) / ((REAL)params.Nxx2);\n", "params.invdx0 = 1.0 / params.dxx0;\n", "params.invdx1 = 1.0 / params.dxx1;\n", "params.invdx2 = 1.0 / params.dxx2;\n", "\\n\"\"\")\n", "\n", "# Generates declare_Cparameters_struct.h, set_Cparameters_default.h, and set_Cparameters[-SIMD].h\n", "par.generate_Cparameters_Ccodes(os.path.join(Ccodesdir))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "# Step 2: `ScalarWave_Playground.c`: The Main C Code \\[Back to [top](#toc)\\]\n", "$$\\label{mainc}$$\n", "\n", "Next we will write the C code infrastructure necessary to make use of the above NRPy+-generated codes. Again, we'll be using RK4 time integration via the Method of Lines." ] }, { "cell_type": "code", "execution_count": 6, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:44.054481Z", "iopub.status.busy": "2021-03-07T17:32:44.053716Z", "iopub.status.idle": "2021-03-07T17:32:44.056916Z", "shell.execute_reply": "2021-03-07T17:32:44.056340Z" } }, "outputs": [], "source": [ "# Part P0: Set the number of ghost cells, from NRPy+'s FD_CENTDERIVS_ORDER\n", "with open(os.path.join(Ccodesdir,\"ScalarWave_NGHOSTS.h\"), \"w\") as file:\n", " file.write(\"// Part P0: Set the number of ghost cells, from NRPy+'s FD_CENTDERIVS_ORDER\\n\")\n", " file.write(\"#define NGHOSTS \"+str(int(par.parval_from_str(\"finite_difference::FD_CENTDERIVS_ORDER\")/2))+\"\\n\")" ] }, { "cell_type": "code", "execution_count": 7, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:44.063546Z", "iopub.status.busy": "2021-03-07T17:32:44.062595Z", "iopub.status.idle": "2021-03-07T17:32:44.066404Z", "shell.execute_reply": "2021-03-07T17:32:44.065774Z" } }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Writing ScalarWave_Ccodes//ScalarWave_Playground.c\n" ] } ], "source": [ "%%writefile $Ccodesdir/ScalarWave_Playground.c\n", "\n", "// Part P0: Import NGHOSTS, which is based on FD_CENTDERIVS_ORDER\n", "#include \"ScalarWave_NGHOSTS.h\"\n", "// Part P0a: set REAL=double, so that all floating point numbers are stored to at least ~16 significant digits.\n", "#define REAL double\n", "\n", "#include \"declare_Cparameters_struct.h\"\n", "\n", "// All SIMD intrinsics used in SIMD-enabled C code loops are defined here:\n", "#include \"../SIMD/SIMD_intrinsics.h\"\n", "\n", "const int NSKIP_0D_OUTPUT = 1;\n", "const int NSKIP_2D_OUTPUT = 10;\n", "\n", "// Part P1: Import needed header files\n", "#include \"stdio.h\"\n", "#include \"stdlib.h\"\n", "#include \"math.h\"\n", "\n", "// Part P2: Add needed #define's to set data type, the IDX4S() macro, and the gridfunctions\n", "// Part P2a: Declare the IDX4S(gf,i,j,k) macro, which enables us to store 4-dimensions of\n", "// data in a 1D array. In this case, consecutive values of \"i\"\n", "// (all other indices held to a fixed value) are consecutive in memory, where\n", "// consecutive values of \"j\" (fixing all other indices) are separated by\n", "// Nxx_plus_2NGHOSTS0 elements in memory. Similarly, consecutive values of\n", "// \"k\" are separated by Nxx_plus_2NGHOSTS0*Nxx_plus_2NGHOSTS1 in memory, etc.\n", "#define IDX4S(g,i,j,k) \\\n", "( (i) + Nxx_plus_2NGHOSTS0 * ( (j) + Nxx_plus_2NGHOSTS1 * ( (k) + Nxx_plus_2NGHOSTS2 * (g) ) ) )\n", "#define LOOP_ALL_GFS_GPS(ii) _Pragma(\"omp parallel for\") \\\n", " for(int (ii)=0;(ii) (Nxx0+2*NGHOSTS)*.25 && i0< (Nxx0+2*NGHOSTS)*.75 &&\n", " i1> (Nxx1+2*NGHOSTS)*.25 && i1< (Nxx1+2*NGHOSTS)*.75) {\n", " const REAL xx0 = xx[0][i0];\n", " const REAL xx1 = xx[1][i1];\n", " REAL uu_exact,vv_exact; exact_solution_single_point(xx0,xx1,xx2,params, &uu_exact,&vv_exact);\n", " fprintf(out2D,\"%e %e %e %e\\n\", xx0,xx1,\n", " numerical_gridfunction_data[IDX4S(0,i0,i1, (int)((Nxx2+ 2*NGHOSTS)*0.5))], uu_exact);\n", " }\n", " }\n", " }\n", " fclose(out2D);\n", "}\n", "\n", "// main() function:\n", "// Step 0: Read command-line input, set up grid structure, allocate memory for gridfunctions, set up coordinates\n", "// Step 1: Set up scalar wave initial data\n", "// Step 2: Evolve scalar wave initial data forward in time using Method of Lines with RK4 algorithm,\n", "// applying quadratic extrapolation outer boundary conditions.\n", "// Step 3: Output relative error between numerical and exact solution.\n", "// Step 4: Free all allocated memory\n", "int main(int argc, const char *argv[]) {\n", " paramstruct params;\n", "#include \"set_Cparameters_default.h\"\n", " // Step 0a: Read command-line input, error out if nonconformant\n", " if(argc != 2 || atoi(argv[1]) < NGHOSTS) {\n", " printf(\"Error: Expected one command-line argument: ./ScalarWave_Playground [Nx(=Ny=Nz)],\\n\");\n", " printf(\"where Nx is the number of grid points in the x,y, and z directions.\\n\");\n", " printf(\"Nx MUST BE larger than NGHOSTS (= %d)\\n\",NGHOSTS);\n", " exit(1);\n", " }\n", " if(atoi(argv[1])%2 != 0) {\n", " printf(\"Error: Algorithm for setting up cell-centered grids here requires Nx, Ny, and Nz to be a multiple of 2 .\\n\");\n", " exit(1);\n", " }\n", "\n", " // Step 0b: Set free parameters, overwriting Cparameters defaults\n", " // by hand or with command-line input, as desired.\n", "#include \"free_parameters.h\"\n", " // ... and then set up the numerical grid structure in time:\n", " const REAL CFL_FACTOR = 0.5; // Set the CFL Factor\n", " #define MIN(A, B) ( ((A) < (B)) ? (A) : (B) )\n", " REAL dt = CFL_FACTOR * MIN(params.dxx0,MIN(params.dxx1,params.dxx2)); // CFL condition\n", "\n", " // Now that params struct has been properly set up, create\n", " // list of const's containing each parameter. E.g.,\n", " // const REAL dxx0 = params.dxx0;\n", "#include \"set_Cparameters-nopointer.h\"\n", "\n", " // Step 0c: Allocate memory for gridfunctions\n", " const int Nxx_plus_2NGHOSTS_tot = Nxx_plus_2NGHOSTS0*Nxx_plus_2NGHOSTS1*Nxx_plus_2NGHOSTS2;\n", " // Step 0d: Allocate memory for gridfunctions\n", "#include \"MoLtimestepping/RK_Allocate_Memory.h\"\n", "\n", " // Step 0e: Set t_final, and number of timesteps based on t_final\n", " const REAL t_final = xxmax[0]*0.8; /* Final time is set so that at t=t_final,\n", " data at the origin have not been corrupted\n", " by the approximate outer boundary condition */\n", " int Nt = (int)(t_final / dt + 0.5); // The number of points in time.\n", " //Add 0.5 to account for C rounding down integers.\n", "\n", " // Step 0f: Set up cell-centered Cartesian coordinate grids\n", " REAL *xx[3];\n", " xx[0] = (REAL *)malloc(sizeof(REAL)*Nxx_plus_2NGHOSTS0);\n", " xx[1] = (REAL *)malloc(sizeof(REAL)*Nxx_plus_2NGHOSTS1);\n", " xx[2] = (REAL *)malloc(sizeof(REAL)*Nxx_plus_2NGHOSTS2);\n", " for(int j=0;j t+dt) in time using\n", " // chosen RK-like MoL timestepping algorithm\n", "#include \"MoLtimestepping/RK_MoL.h\"\n", " } // End main loop to progress forward in time.\n", "\n", " // Step 4: Free all allocated memory\n", "#include \"MoLtimestepping/RK_Free_Memory.h\"\n", " for(int i=0;i<3;i++) free(xx[i]);\n", " return 0;\n", "}" ] }, { "cell_type": "code", "execution_count": 8, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:44.075262Z", "iopub.status.busy": "2021-03-07T17:32:44.072124Z", "iopub.status.idle": "2021-03-07T17:32:49.224157Z", "shell.execute_reply": "2021-03-07T17:32:49.223317Z" }, "scrolled": true }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Compiling executable...\n", "(EXEC): Executing `gcc -std=gnu99 -Ofast -fopenmp -march=native -funroll-loops ScalarWave_Ccodes/ScalarWave_Playground.c -o ScalarWave_Ccodes/output/ScalarWave_Playground -lm`...\n", "(BENCH): Finished executing in 0.6068646907806396 seconds.\n", "Finished compilation.\n", "(EXEC): Executing `taskset -c 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15 ./ScalarWave_Playground 48`...\n", "(BENCH): Finished executing in 0.2051081657409668 seconds.\n", "(EXEC): Executing `taskset -c 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15 ./ScalarWave_Playground 64`...\n", "(BENCH): Finished executing in 0.20606541633605957 seconds.\n", "(EXEC): Executing `taskset -c 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15 ./ScalarWave_Playground 96`...\n", "(BENCH): Finished executing in 1.0081427097320557 seconds.\n" ] } ], "source": [ "cmd.C_compile(os.path.join(Ccodesdir,\"ScalarWave_Playground.c\"),\n", " os.path.join(outdir,\"ScalarWave_Playground\"))\n", "#!icc -align -qopenmp -xHost -O2 -qopt-report=5 -qopt-report-phase ipo -qopt-report-phase vec -vec-threshold1 -qopt-prefetch=4 ScalarWave/ScalarWave_Playground.c -o ScalarWave_Playground\n", "\n", "# 10o FD testing:\n", "# 4.46s\n", "# !icc -align -qopenmp -xHost -O2 -qopt-report=5 -qopt-report-phase ipo -qopt-report-phase vec -vec-threshold1 -qopt-prefetch=4 ScalarWave/ScalarWave_Playground.c -o ScalarWave_Playground\n", "# 4.65s\n", "# !gcc -Ofast -fopenmp -march=native ScalarWave/ScalarWave_Playground.c -fopt-info-vec-optimized-missed -o ScalarWave_Playground -lm 2>&1 |grep RHS\n", "# 5.45s\n", "# !clang -Ofast -fopenmp -mavx2 -mfma ScalarWave/ScalarWave_Playground.c -o ScalarWave_Playground -lm\n", "\n", "# Change to output directory\n", "os.chdir(outdir)\n", "# Clean up existing output files\n", "cmd.delete_existing_files(\"out??.txt\")\n", "cmd.Execute(\"ScalarWave_Playground\", \"48\", \"out48.txt\")\n", "cmd.Execute(\"ScalarWave_Playground\", \"64\", \"out64.txt\")\n", "cmd.Execute(\"ScalarWave_Playground\", \"96\", \"out96.txt\")\n", "# for benchmarking:\n", "# %timeit cmd.Execute(\"ScalarWave_Playground\", \"148\", \"out148.txt\", verbose=False)\n", "# FD functions enabled\n", "# 7.06 s ± 2.27 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n", "# disabled:\n", "# 7.06 s ± 2.58 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n", "\n", "# Return to root directory\n", "os.chdir(os.path.join(\"../../\"))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "# Step 3: Code Validation: Verify that relative error in numerical solution converges to zero at the expected order \\[Back to [top](#toc)\\]\n", "$$\\label{convergence}$$" ] }, { "cell_type": "code", "execution_count": 9, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:49.240042Z", "iopub.status.busy": "2021-03-07T17:32:49.238337Z", "iopub.status.idle": "2021-03-07T17:32:49.808041Z", "shell.execute_reply": "2021-03-07T17:32:49.808549Z" } }, "outputs": [ { "data": { "image/png": "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\n", "text/plain": [ "
" ] }, "metadata": { "needs_background": "light" }, "output_type": "display_data" } ], "source": [ "%matplotlib inline\n", "import matplotlib.pyplot as plt\n", "import mpmath as mp\n", "import csv\n", "\n", "def file_reader(filename):\n", " with open(filename) as file:\n", " reader = csv.reader(file, delimiter=\" \")\n", " data = list(zip(*reader))\n", " # data is a tuple of strings. Tuples are immutable, and we need to perform math on\n", " # the data, so here we convert tuple to lists of floats:\n", " data0 = []\n", " data1 = []\n", " for i in range(len(data[0])):\n", " data0.append(float(data[0][i]))\n", " data1.append(float(data[1][i]))\n", " return data0,data1\n", "\n", "first_col48,second_col48 = file_reader(os.path.join(outdir,\"out48.txt\"))\n", "first_col64,second_col64 = file_reader(os.path.join(outdir,\"out64.txt\"))\n", "first_col96,second_col96 = file_reader(os.path.join(outdir,\"out96.txt\"))\n", "\n", "for i in range(len(second_col64)):\n", " # data64 = data48*(64/48)**4\n", " # -> log10(data64) = log10(data48) + 4*log(64/48)\n", " second_col64[i] += 4*mp.log10(64./48.)\n", "for i in range(len(second_col96)):\n", " # data96 = data48*(96/48)**4\n", " # -> log10(data96) = log10(data48) + 4*log(96/48)\n", " second_col96[i] += 4*mp.log10(96./48.)\n", "\n", "# https://matplotlib.org/gallery/text_labels_and_annotations/legend.html#sphx-glr-gallery-text-labels-and-annotations-legend-py\n", "fig, ax = plt.subplots()\n", "\n", "plt.title(\"Plot Demonstrating 4th-order Convergence\")\n", "plt.xlabel(\"time\")\n", "plt.ylabel(\"log10(Relative error)\")\n", "\n", "ax.plot(first_col48, second_col48, 'k--', label='Nx = 48')\n", "ax.plot(first_col64, second_col64, 'k-', label='Nx = 64, mult by (64/48)^4')\n", "ax.plot(first_col96, second_col96, 'k.', label='Nx = 96, mult by (96/48)^4')\n", "\n", "legend = ax.legend(loc='lower right', shadow=True, fontsize='x-large')\n", "legend.get_frame().set_facecolor('C1')\n", "plt.show()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "\n", "\n", "# Step 4: Output this notebook to $\\LaTeX$-formatted PDF file \\[Back to [top](#toc)\\]\n", "$$\\label{latex_pdf_output}$$\n", "\n", "The following code cell converts this Jupyter notebook into a proper, clickable $\\LaTeX$-formatted PDF file. After the cell is successfully run, the generated PDF may be found in the root NRPy+ tutorial directory, with filename\n", "[Tutorial-Start_to_Finish-ScalarWave.pdf](Tutorial-Start_to_Finish-ScalarWave.pdf) (Note that clicking on this link may not work; you may need to open the PDF file through another means.)" ] }, { "cell_type": "code", "execution_count": 10, "metadata": { "execution": { "iopub.execute_input": "2021-03-07T17:32:49.815973Z", "iopub.status.busy": "2021-03-07T17:32:49.812835Z", "iopub.status.idle": "2021-03-07T17:32:53.918708Z", "shell.execute_reply": "2021-03-07T17:32:53.917654Z" } }, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Created Tutorial-Start_to_Finish-ScalarWave.tex, and compiled LaTeX file to\n", " PDF file Tutorial-Start_to_Finish-ScalarWave.pdf\n" ] } ], "source": [ "import cmdline_helper as cmd # NRPy+: Multi-platform Python command-line interface\n", "cmd.output_Jupyter_notebook_to_LaTeXed_PDF(\"Tutorial-Start_to_Finish-ScalarWave\")" ] } ], "metadata": { "kernelspec": { "display_name": "Python 3 (ipykernel)", "language": "python", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.10.0rc2" } }, "nbformat": 4, "nbformat_minor": 2 }