;;; GNU Guix --- Functional package management for GNU ;;; Copyright © 2017, 2018 Paul Garlick ;;; ;;; This file is part of GNU Guix. ;;; ;;; GNU Guix is free software; you can redistribute it and/or modify it ;;; under the terms of the GNU General Public License as published by ;;; the Free Software Foundation; either version 3 of the License, or (at ;;; your option) any later version. ;;; ;;; GNU Guix is distributed in the hope that it will be useful, but ;;; WITHOUT ANY WARRANTY; without even the implied warranty of ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ;;; GNU General Public License for more details. ;;; ;;; You should have received a copy of the GNU General Public License ;;; along with GNU Guix. If not, see . (define-module (gnu packages simulation) #:use-module (gnu packages) #:use-module (gnu packages base) #:use-module (gnu packages bash) #:use-module (gnu packages bison) #:use-module (gnu packages boost) #:use-module (gnu packages check) #:use-module (gnu packages compression) #:use-module (gnu packages flex) #:use-module (gnu packages gettext) #:use-module (gnu packages gcc) #:use-module (gnu packages gl) #:use-module (gnu packages graphics) #:use-module (gnu packages gtk) #:use-module (gnu packages linux) #:use-module (gnu packages m4) #:use-module (gnu packages maths) #:use-module (gnu packages mpi) #:use-module (gnu packages multiprecision) #:use-module (gnu packages ncurses) #:use-module (gnu packages python) #:use-module (gnu packages readline) #:use-module (gnu packages ssh) #:use-module (gnu packages tls) #:use-module (gnu packages version-control) #:use-module (gnu packages xml) #:use-module (gnu packages xorg) #:use-module (guix download) #:use-module (guix build utils) #:use-module (guix build-system gnu) #:use-module (guix build-system python) #:use-module ((guix licenses) #:prefix license:) #:use-module (guix packages) #:use-module (guix utils) #:use-module (ice-9 ftw) #:use-module (ice-9 regex) #:use-module (srfi srfi-1)) (define-public openfoam (package (name "openfoam") (version "4.1") (source (origin (method url-fetch) (uri (string-append "http://dl.openfoam.org/source/" (string-map (lambda (x) (if (eq? x #\.) #\- x)) version))) (file-name (string-append name "-" version ".tar.gz")) (sha256 (base32 "0cgxh4h2hf50qbvvdg5miwc2nympb0nrv3md96vb3gbs9vk8vq9d")) (patches (search-patches "openfoam-4.1-cleanup.patch")))) (build-system gnu-build-system) (inputs `(("boost" ,boost) ("cgal" ,cgal) ("flex" ,flex) ("git" ,git) ("gmp" ,gmp) ("libxt" ,libxt) ("metis" ,metis) ("mpfr" ,mpfr) ("ncurses" ,ncurses) ("readline" ,readline) ("scotch" ,pt-scotch32) ("zlib" ,zlib))) (native-inputs `(("bison" ,bison))) (propagated-inputs `(("gzip" ,gzip) ("gnuplot" ,gnuplot) ("openmpi" ,openmpi))) (outputs '("debug" ;~60MB "out")) (arguments `( ;; Executable files and shared libraries are located in the 'platforms' ;; subdirectory. #:strip-directories (list (string-append "lib/OpenFOAM-" ,version "/platforms/linux64GccDPInt32Opt/bin") (string-append "lib/OpenFOAM-" ,version "/platforms/linux64GccDPInt32Opt/lib")) #:tests? #f ; no tests to run #:modules ((ice-9 ftw) (ice-9 regex) (guix build gnu-build-system) (guix build utils)) #:phases (modify-phases %standard-phases (add-after 'unpack 'rename-build-directory (lambda _ (chdir "..") ;; Use 'OpenFOAM-version' convention to match the file ;; name expectations in the build phase. (let ((unpack-dir (string-append (getcwd) "/" (list-ref (scandir (getcwd) (lambda (name) (string-match "^OpenFOAM" name))) 0))) (build-dir (string-append (getcwd) "/OpenFOAM-" ,version))) (rename-file unpack-dir build-dir) ; rename build directory (chdir (basename build-dir))) ; move to build directory #t)) (delete 'configure) ; no configure phase (replace 'build (lambda _ (let ((libraries '("boost" "cgal" "gmp" "metis" "mpfr" "scotch"))) ;; set variables to define store paths (for-each (lambda (library) (setenv (string-append (string-upcase library) "_ROOT") (assoc-ref %build-inputs library))) libraries)) ;; set variables to define package versions (setenv "SCOTCHVERSION" ,(package-version scotch)) (setenv "METISVERSION" ,(package-version metis)) ;; set variable to pass extra 'rpath' arguments to linker (setenv "LDFLAGS" (string-append "-Wl," "-rpath=" %output "/lib/OpenFOAM-" ,version "/platforms/linux64GccDPInt32Opt/lib," "-rpath=" %output "/lib/OpenFOAM-" ,version "/platforms/linux64GccDPInt32Opt/lib/dummy")) ;; compile OpenFOAM libraries and applications (zero? (system (format #f "source ./etc/bashrc && ./Allwmake -j~a" (parallel-job-count)))))) (add-after 'build 'update-configuration-files (lambda _ ;; record store paths and package versions in ;; configuration files (substitute* "etc/config.sh/CGAL" (("$BOOST_ROOT") (getenv "BOOST_ROOT"))) (substitute* "etc/config.sh/CGAL" (("$CGAL_ROOT") (getenv "CGAL_ROOT"))) (substitute* "etc/config.sh/metis" (("$METIS_ROOT") (getenv "METIS_ROOT"))) (substitute* "etc/config.sh/metis" (("$METISVERSION") (getenv "METISVERSION"))) (substitute* "etc/config.sh/scotch" (("$SCOTCH_ROOT") (getenv "SCOTCH_ROOT"))) (substitute* "etc/config.sh/scotch" (("$SCOTCHVERSION") (getenv "SCOTCHVERSION"))) (substitute* "etc/config.sh/settings" (("$GMP_ROOT") (getenv "GMP_ROOT"))) (substitute* "etc/config.sh/settings" (("$MPFR_ROOT") (getenv "MPFR_ROOT"))) ;; reset lockDir variable to refer to write-enabled ;; directory (substitute* "wmake/wmake" ((" lockDir=.*$") " lockDir=$HOME/.$WM_PROJECT/.wmake\n")) (substitute* "wmake/wmakeScheduler" (("lockDir=.*$") "lockDir=$HOME/.$WM_PROJECT/.wmake\n")) (substitute* "wmake/wmakeSchedulerUptime" (("lockDir=.*$") "lockDir=$HOME/.$WM_PROJECT/.wmake\n")) #t)) (add-after 'build 'cleanup ;; Avoid unncessary, voluminous object and dep files. (lambda _ (delete-file-recursively "platforms/linux64GccDPInt32Opt/src") (delete-file-recursively "platforms/linux64GccDPInt32OptSYSTEMOPENMPI") (for-each delete-file (find-files "." "\\.o$")) #t)) (replace 'install (lambda _ ;; use 'OpenFOAM-version' convention (let ((install-dir (string-append %output "/lib/OpenFOAM-" ,version))) (mkdir-p install-dir) ; create install directory ;; move contents of build directory to install directory (copy-recursively "." install-dir)))) (add-after 'install 'add-symbolic-link (lambda _ ;; add symbolic link for standard 'bin' directory (symlink (string-append "./lib/OpenFOAM-" ,version "/platforms/linux64GccDPInt32Opt/bin") (string-append %output "/bin")) #t))))) ;; Note: ;; Tutorial files are installed read-only in /gnu/store. ;; To allow write permissions on files copied from the store a ;; 'chmod' step is needed before running the applications. For ;; example, from a user's login: ;; $ source $GUIX_PROFILE/lib/OpenFOAM-4.1/etc/bashrc ;; $ mkdir -p $FOAM_RUN ;; $ cd $FOAM_RUN ;; $ cp -r $FOAM_TUTORIALS/incompressible/simpleFoam/pitzDaily . ;; $ cd pitzDaily ;; $ chmod -R u+w . ;; $ blockMesh (synopsis "Framework for numerical simulation of fluid flow") (description "OpenFOAM provides a set of solvers and methods for tackling problems in the field of Computational Fluid Dynamics (CFD). It is written in C++. Governing equations such as the Navier-Stokes equations can be solved in integral form. Physical processes such as phase change, droplet transport and chemical reaction can be modelled. Numerical methods are included to deal with sharp gradients, such as those encountered in flows with shock waves and flows with gas/liquid interfaces. Large problems may be split into smaller, connected problems for efficient solution on parallel systems.") (license license:gpl3+) (home-page "https://openfoam.org"))) (define-public python-mpi4py (package (name "python-mpi4py") (version "3.0.0") (source (origin (method url-fetch) (uri (pypi-uri "mpi4py" version)) (sha256 (base32 "1mzgd26dfv4vwbci8gq77ss9f0x26i9aqzq9b9vs9ndxhlnv0mxl")))) (build-system python-build-system) (arguments `(#:phases (modify-phases %standard-phases (add-after 'build 'mpi-setup ,%openmpi-setup) (add-before 'check 'pre-check (lambda _ ;; Skip BaseTestSpawn class (causes error 'ompi_dpm_dyn_init() ;; failed --> Returned "Unreachable"' in chroot environment). (substitute* "test/test_spawn.py" (("unittest.skipMPI\\('openmpi\\(<3.0.0\\)'\\)") "unittest.skipMPI('openmpi')")) #t))))) (inputs `(("openmpi" ,openmpi))) (home-page "https://bitbucket.org/mpi4py/mpi4py/") (synopsis "Python bindings for the Message Passing Interface standard") (description "MPI for Python (mpi4py) provides bindings of the Message Passing Interface (MPI) standard for the Python programming language, allowing any Python program to exploit multiple processors. mpi4py is constructed on top of the MPI-1/MPI-2 specification and provides an object oriented interface which closely follows MPI-2 C++ bindings. It supports point-to-point and collective communications of any picklable Python object as well as optimized communications of Python objects (such as NumPy arrays) that expose a buffer interface.") (license license:bsd-3))) (define-public python-petsc4py (package (name "python-petsc4py") (version "3.9.1") (source (origin (method url-fetch) (uri (pypi-uri "petsc4py" version)) (sha256 (base32 "1f8zd1ac9irsgkyqmzq30d9kl10fy1nh6zk312dhs43g449fkkhc")))) (build-system python-build-system) (arguments `(#:phases (modify-phases %standard-phases (add-before 'build 'pre-build (lambda _ ;; Define path to PETSc installation. (setenv "PETSC_DIR" (assoc-ref %build-inputs "petsc")) #t)) (add-before 'check 'mpi-setup ,%openmpi-setup)))) (inputs `(("petsc" ,petsc-openmpi) ("python-numpy" ,python-numpy))) (home-page "https://bitbucket.org/petsc/petsc4py/") (synopsis "Python bindings for PETSc") (description "PETSc, the Portable, Extensible Toolkit for Scientific Computation, is a suite of data structures and routines for the scalable (parallel) solution of scientific applications modeled by partial differential equations. It employs the MPI standard for all message-passing communication. @code{petsc4py} provides Python bindings to almost all functions of PETSc.") (license license:bsd-3))) (define-public python-slepc4py (package (name "python-slepc4py") (version "3.9.0") (source (origin (method url-fetch) (uri (pypi-uri "slepc4py" version)) (sha256 (base32 "02xr0vndgibgkz3rgprqk05n3mk5mpgqw550sr4681vcsgz4zvb7")))) (build-system python-build-system) (arguments `(#:phases (modify-phases %standard-phases (add-before 'build 'pre-build (lambda _ ;; Define path to PETSc installation. (setenv "PETSC_DIR" (assoc-ref %build-inputs "petsc")) ;; Define path to SLEPc installation. (setenv "SLEPC_DIR" (assoc-ref %build-inputs "slepc")) #t)) (add-before 'check 'mpi-setup ,%openmpi-setup)))) (inputs `(("python-numpy" ,python-numpy) ("python-petsc4py" ,python-petsc4py) ("slepc" ,slepc-openmpi))) (home-page "https://bitbucket.org/slepc/slepc4py/") (synopsis "Python bindings for SLEPc") (description "SLEPc, the Scalable Library for Eigenvalue Problem Computations, is based on PETSc, the Portable, Extensible Toolkit for Scientific Computation. It employs the MPI standard for all message-passing communication. @code{slepc4py} provides Python bindings to almost all functions of SLEPc.") (license license:bsd-3))) (define-public python-fenics-dijitso (package (name "python-fenics-dijitso") (version "2018.1.0") (source (origin (method url-fetch) (uri (pypi-uri "fenics-dijitso" version)) (sha256 (base32 "1qax2f52qsjbd1h5lk5i5shp448qlakxabjjybrfc1w823p0yql9")))) (build-system python-build-system) (inputs `(("openmpi" ,openmpi) ("python-numpy" ,python-numpy))) (native-inputs `(("python-pytest-cov" ,python-pytest-cov))) (propagated-inputs `(("python-mpi4py" ,python-mpi4py))) (arguments `(#:phases (modify-phases %standard-phases (add-after 'build 'mpi-setup ,%openmpi-setup) (replace 'check (lambda _ (setenv "HOME" "/tmp") (setenv "PYTHONPATH" (string-append (getcwd) ":" (getenv "PYTHONPATH"))) (chdir "test") (invoke "./runtests.sh") (chdir "..") #t))))) (home-page "https://bitbucket.org/fenics-project/dijitso/") (synopsis "Distributed just-in-time building of shared libraries") (description "Dijitso provides a core component of the @code{FEniCS} framework, namely the just-in-time compilation of C++ code that is generated from Python modules. It is called from within a C++ library, using ctypes to import the dynamic shared library directly. As long as the compiled code can provide a simple factory function to a class implementing a predefined C++ interface, there is no limit to the complexity of that interface. Parallel support depends on the @code{mpi4py} interface.") (license license:lgpl3+)))