.. DO NOT EDIT. .. THIS FILE WAS AUTOMATICALLY GENERATED BY SPHINX-GALLERY. .. TO MAKE CHANGES, EDIT THE SOURCE PYTHON FILE: .. "content/examples/06-tdem/plot_fwd_tdem_3d_model.py" .. LINE NUMBERS ARE GIVEN BELOW. .. only:: html .. note:: :class: sphx-glr-download-link-note :ref:`Go to the end ` to download the full example code .. rst-class:: sphx-glr-example-title .. _sphx_glr_content_examples_06-tdem_plot_fwd_tdem_3d_model.py: Time-domain CSEM for a resistive cube in a deep marine setting ============================================================== .. GENERATED FROM PYTHON SOURCE LINES 5-18 .. code-block:: default import empymod import discretize try: from pymatsolver import Pardiso as Solver except ImportError: from SimPEG import SolverLU as Solver import numpy as np from SimPEG import maps from SimPEG.electromagnetics import time_domain as TDEM import matplotlib.pyplot as plt .. GENERATED FROM PYTHON SOURCE LINES 19-21 (A) Model --------- .. GENERATED FROM PYTHON SOURCE LINES 21-77 .. code-block:: default fig = plt.figure(figsize=(5.5, 3)) ax = plt.gca() # Seafloor and background plt.plot([-200, 2200], [-2000, -2000], "-", c=".4") bg = plt.Rectangle((-500, -3000), 3000, 1000, facecolor="black", alpha=0.1) ax.add_patch(bg) # Plot survey plt.plot([-50, 50], [-1950, -1950], "*-", ms=8, c="k") plt.plot(2000, -2000, "v", ms=8, c="k") plt.text(0, -1900, r"Tx", horizontalalignment="center") plt.text(2000, -1900, r"Rx", horizontalalignment="center") # Plot cube plt.plot([450, 1550, 1550, 450, 450], [-2300, -2300, -2700, -2700, -2300], "k-") plt.plot([300, 1400, 1400, 300, 300], [-2350, -2350, -2750, -2750, -2350], "k:") plt.plot([600, 600, 1700, 1700, 1550], [-2300, -2250, -2250, -2650, -2650], "k:") plt.plot([300, 600], [-2350, -2250], "k:") plt.plot([1400, 1700], [-2350, -2250], "k:") plt.plot([300, 450], [-2750, -2700], "k:") plt.plot([1400, 1700], [-2750, -2650], "k:") tg = plt.Rectangle((450, -2700), 1100, 400, facecolor="black", alpha=0.2) ax.add_patch(tg) # Annotate resistivities plt.text( 1000, -1925, r"$\rho_\mathrm{sea}=0.3\,\Omega\,$m", horizontalalignment="center" ) plt.text( 1000, -2150, r"$\rho_\mathrm{bg}=1.0\,\Omega\,$m", horizontalalignment="center" ) plt.text( 1000, -2550, r"$\rho_\mathrm{tg}=100.0\,\Omega\,$m", horizontalalignment="center" ) plt.text(1500, -2800, r"$y=-500\,$m", horizontalalignment="left") plt.text(1750, -2650, r"$y=500\,$m", horizontalalignment="left") # Ticks and labels plt.xticks( [-50, 50, 450, 1550, 2000], ["$-50~$ $~$ $~$", " $~50$", "$450$", "$1550$", "$2000$"], ) plt.yticks( [-1950, -2000, -2300, -2700], ["$-1950$\n", "\n$-2000$", "$-2300$", "$-2700$"] ) plt.xlim([-200, 2200]) plt.ylim([-3000, -1800]) plt.xlabel("$x$ (m)") plt.ylabel("$z$ (m)") plt.tight_layout() plt.show() .. image-sg:: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_001.png :alt: plot fwd tdem 3d model :srcset: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_001.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 78-93 .. code-block:: default # Resistivities res_sea = 0.3 res_bg = 1.0 res_tg = 100.0 # Seafloor seafloor = -2000 # Target dimension tg_x = [450, 1550] tg_y = [-500, 500] tg_z = [-2700, -2300] .. GENERATED FROM PYTHON SOURCE LINES 94-96 (B) Survey ---------- .. GENERATED FROM PYTHON SOURCE LINES 96-109 .. code-block:: default # Source: 100 m x-directed diplole at the origin, # 50 m above seafloor, src [x1, x2, y1, y2, z1, z2] src = [-50, 50, 0, 0, -1950, -1950] # Receiver: x-directed dipole at 2 km on the # seafloor, rec = [x, y, z, azimuth, dip] rec = [2000, 0, -2000, 0, 0] # Times to compute, 0.1 - 10 s, 301 steps times = np.logspace(-1, 1, 301) .. GENERATED FROM PYTHON SOURCE LINES 110-115 (C) Modelling parameters ------------------------ Check diffusion distances """"""""""""""""""""""""" .. GENERATED FROM PYTHON SOURCE LINES 115-126 .. code-block:: default # Get min/max diffusion distances for the two halfspaces. diff_dist0 = 1261 * np.sqrt(np.r_[times * res_sea, times * res_sea]) diff_dist1 = 1261 * np.sqrt(np.r_[times * res_bg, times * res_bg]) diff_dist2 = 1261 * np.sqrt(np.r_[times * res_tg, times * res_tg]) print("Min/max diffusion distance:") print(f"- Water :: {diff_dist0.min():8.0f} / {diff_dist0.max():8.0f} m.") print(f"- Background :: {diff_dist1.min():8.0f} / {diff_dist1.max():8.0f} m.") print(f"- Target :: {diff_dist2.min():8.0f} / {diff_dist2.max():8.0f} m.") .. rst-class:: sphx-glr-script-out .. code-block:: none Min/max diffusion distance: - Water :: 218 / 2184 m. - Background :: 399 / 3988 m. - Target :: 3988 / 39876 m. .. GENERATED FROM PYTHON SOURCE LINES 127-129 Time-steps """""""""" .. GENERATED FROM PYTHON SOURCE LINES 129-166 .. code-block:: default # Time steps time_steps = [1e-1, (1e-2, 21), (3e-2, 23), (1e-1, 21), (3e-1, 23)] # Create mesh with time steps ts = discretize.TensorMesh([time_steps]).nodes_x # Plot them plt.figure(figsize=(9, 1.5)) # Logarithmic scale plt.subplot(121) plt.title("Check time-steps on logarithmic-scale") plt.plot([times.min(), times.min()], [-1, 1]) plt.plot([times.max(), times.max()], [-1, 1]) plt.plot(ts, ts * 0, ".", ms=2) plt.yticks([]) plt.xscale("log") plt.xlabel("Time (s)") # Linear scale plt.subplot(122) plt.title("Check time-steps on linear-scale") plt.plot([times.min(), times.min()], [-1, 1]) plt.plot([times.max(), times.max()], [-1, 1]) plt.plot(ts, ts * 0, ".", ms=2) plt.yticks([]) plt.xlabel("Time (s)") plt.tight_layout() plt.show() # Check times with time-steps print(f"Min/max times : {times.min():.1e} / {times.max():.1e}") print(f"Min/max timeSteps: {ts[1]:.1e} / {ts[-1]:.1e}") .. image-sg:: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_002.png :alt: Check time-steps on logarithmic-scale, Check time-steps on linear-scale :srcset: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_002.png :class: sphx-glr-single-img .. rst-class:: sphx-glr-script-out .. code-block:: none Min/max times : 1.0e-01 / 1.0e+01 Min/max timeSteps: 1.0e-01 / 1.0e+01 .. GENERATED FROM PYTHON SOURCE LINES 167-169 Create mesh (`discretize`) """""""""""""""""""""""""" .. GENERATED FROM PYTHON SOURCE LINES 169-201 .. code-block:: default # Cell width, number of cells width = 100 nx = rec[0] // width + 4 ny = 10 nz = 9 # Padding npadx = 14 npadyz = 12 # Stretching alpha = 1.3 # Initiate TensorMesh mesh = discretize.TensorMesh( [ [(width, npadx, -alpha), (width, nx), (width, npadx, alpha)], [(width, npadyz, -alpha), (width, ny), (width, npadyz, alpha)], [(width, npadyz, -alpha), (width, nz), (width, npadyz, alpha)], ], x0="CCC", ) # Shift mesh so that # x=0 is at midpoint of source; # z=-2000 is at receiver level mesh.x0[0] += rec[0] // 2 - width / 2 mesh.x0[2] -= nz / 2 * width - seafloor mesh .. raw:: html
TensorMesh 58,344 cells
MESH EXTENT CELL WIDTH FACTOR
dir nC min max min max max
x 52 -16,878.63 18,778.63 100.00 3,937.38 1.30
y 34 -10,162.50 10,162.50 100.00 2,329.81 1.30
z 33 -12,562.50 7,662.50 100.00 2,329.81 1.30


.. GENERATED FROM PYTHON SOURCE LINES 202-207 Check if source and receiver are exactly at x-edges. """""""""""""""""""""""""""""""""""""""""""""""""""" No requirement; if receiver are exactly on x-edges then no interpolation is required to get the responses (cell centers in x, cell edges in y, z). .. GENERATED FROM PYTHON SOURCE LINES 207-227 .. code-block:: default print( f"Rec-{{x;y;z}} :: {rec[0] in np.round(mesh.cell_centers_x)!s:>5}; " f"{rec[1] in np.round(mesh.nodes_y)!s:>5}; " f"{rec[2] in np.round(mesh.nodes_z)!s:>5}" ) print( f"Src-x :: {src[0] in np.round(mesh.cell_centers_x)!s:>5}; " f"{src[1] in np.round(mesh.cell_centers_x)!s:>5}" ) print( f"Src-y :: {src[2] in np.round(mesh.nodes_y)!s:>5}; " f"{src[3] in np.round(mesh.nodes_y)!s:>5}" ) print( f"Src-z :: {src[4] in np.round(mesh.nodes_z)!s:>5}; " f"{src[5] in np.round(mesh.nodes_z)!s:>5}" ) .. rst-class:: sphx-glr-script-out .. code-block:: none Rec-{x;y;z} :: True; True; True Src-x :: False; False Src-y :: True; True Src-z :: False; False .. GENERATED FROM PYTHON SOURCE LINES 228-230 Put model on mesh """"""""""""""""" .. GENERATED FROM PYTHON SOURCE LINES 230-257 .. code-block:: default # Background model mres_bg = np.ones(mesh.nC) * res_sea # Upper halfspace; sea water mres_bg[mesh.gridCC[:, 2] < seafloor] = res_bg # Lower halfspace; background # Target model mres_tg = mres_bg.copy() # Copy background model target_inds = ( # Find target indices (mesh.gridCC[:, 0] >= tg_x[0]) & (mesh.gridCC[:, 0] <= tg_x[1]) & (mesh.gridCC[:, 1] >= tg_y[0]) & (mesh.gridCC[:, 1] <= tg_y[1]) & (mesh.gridCC[:, 2] >= tg_z[0]) & (mesh.gridCC[:, 2] <= tg_z[1]) ) mres_tg[target_inds] = res_tg # Target resistivity # QC mesh.plot_3d_slicer( np.log10(mres_tg), clim=[np.log10(res_sea), np.log10(res_tg)], xlim=[-src[0] - 100, rec[0] + 100], ylim=[-rec[0] / 2, rec[0] / 2], zlim=[tg_z[0] - 100, seafloor + 100], ) .. image-sg:: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_003.png :alt: plot fwd tdem 3d model :srcset: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_003.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 258-263 (D) `empymod` ------------- Compute the 1D background semi-analytically, using 5 points to approximate the 100-m long dipole. .. GENERATED FROM PYTHON SOURCE LINES 263-279 .. code-block:: default inp = { "src": src.copy(), "rec": rec.copy(), "depth": seafloor, "res": [res_sea, res_bg], "freqtime": times, "signal": -1, # Switch-off "srcpts": 5, # 5 points for finite length approx "strength": 1, # Account for source length "verb": 1, } epm_bg = empymod.bipole(**inp) .. GENERATED FROM PYTHON SOURCE LINES 280-284 (E) `SimPEG` ------------ Set-up SimPEG-specific parameters. .. GENERATED FROM PYTHON SOURCE LINES 284-323 .. code-block:: default # Set up the receiver list rec_list = [ TDEM.Rx.PointElectricField( orientation="x", times=times, locations=np.array( [ [*rec[:3]], ] ), ), ] # Set up the source list src_list = [ TDEM.Src.LineCurrent( receiver_list=rec_list, location=np.array([[*src[::2]], [*src[1::2]]]), ), ] # Create `Survey` survey = TDEM.Survey(src_list) # Define the `Simulation` prob = TDEM.Simulation3DElectricField( mesh, survey=survey, rhoMap=maps.IdentityMap(mesh), solver=Solver, time_steps=time_steps, ) .. GENERATED FROM PYTHON SOURCE LINES 324-326 Compute """"""" .. GENERATED FROM PYTHON SOURCE LINES 326-331 .. code-block:: default spg_bg = prob.dpred(mres_bg) spg_tg = prob.dpred(mres_tg) .. GENERATED FROM PYTHON SOURCE LINES 332-334 (F) Plots --------- .. GENERATED FROM PYTHON SOURCE LINES 334-361 .. code-block:: default plt.figure(figsize=(5, 4)) ax1 = plt.subplot(111) plt.title("Resistive cube in a deep marine setting") plt.plot(times, epm_bg * 1e9, ".4", lw=2, label="empymod") plt.plot(times, spg_bg * 1e9, "C0--", label="SimPEG Background") plt.plot(times, spg_tg * 1e9, "C1--", label="SimPEG Target") plt.ylabel("$E_x$ (nV/m)") plt.xscale("log") plt.xlim([0.1, 10]) plt.legend(loc=3) plt.grid(axis="y", c="0.9") plt.xlabel("Time (s)") # Switch off spines ax1.spines["top"].set_visible(False) ax1.spines["right"].set_visible(False) plt.tight_layout() plt.show() .. image-sg:: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_004.png :alt: Resistive cube in a deep marine setting :srcset: /content/examples/06-tdem/images/sphx_glr_plot_fwd_tdem_3d_model_004.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 362-363 empymod.Report([SimPEG, discretize, pymatsolver]) .. rst-class:: sphx-glr-timing **Total running time of the script:** ( 4 minutes 50.711 seconds) **Estimated memory usage:** 1909 MB .. _sphx_glr_download_content_examples_06-tdem_plot_fwd_tdem_3d_model.py: .. only:: html .. container:: sphx-glr-footer sphx-glr-footer-example .. container:: sphx-glr-download sphx-glr-download-python :download:`Download Python source code: plot_fwd_tdem_3d_model.py ` .. container:: sphx-glr-download sphx-glr-download-jupyter :download:`Download Jupyter notebook: plot_fwd_tdem_3d_model.ipynb ` .. only:: html .. rst-class:: sphx-glr-signature `Gallery generated by Sphinx-Gallery `_