2020-08-29 08:56:15 +08:00
|
|
|
! -------------------------------------------------------------------------
|
|
|
|
! LAMMPS - Large-scale Atomic/Molecular Massively Parallel Simulator
|
2021-05-25 04:16:21 +08:00
|
|
|
! https://www.lammps.org/ Sandia National Laboratories
|
2022-10-24 23:06:32 +08:00
|
|
|
! LAMMPS Development team: developers@lammps.org
|
2020-08-29 08:56:15 +08:00
|
|
|
!
|
|
|
|
! Copyright (2003) Sandia Corporation. Under the terms of Contract
|
|
|
|
! DE-AC04-94AL85000 with Sandia Corporation, the U.S. Government retains
|
|
|
|
! certain rights in this software. This software is distributed under
|
|
|
|
! the GNU General Public License.
|
|
|
|
!
|
|
|
|
! See the README file in the top-level LAMMPS directory.
|
|
|
|
! -------------------------------------------------------------------------
|
|
|
|
!
|
|
|
|
! Fortran interface to the LAMMPS library implemented as a Fortran 2003
|
|
|
|
! style module that wraps the C-style library interface in library.cpp
|
|
|
|
! and library.h using the ISO_C_BINDING module of the Fortran compiler.
|
|
|
|
!
|
|
|
|
! Based on the LAMMPS Fortran 2003 module contributed by:
|
2020-10-02 22:36:17 +08:00
|
|
|
! Karl D. Hammond <hammondkd@missouri.edu>
|
|
|
|
! University of Missouri, 2012-2020
|
2020-08-29 08:56:15 +08:00
|
|
|
!
|
2022-10-24 23:06:32 +08:00
|
|
|
! Contributing authors:
|
2023-06-20 03:28:27 +08:00
|
|
|
! - Axel Kohlmeyer <akohlmey@gmail.com>, Temple University, 2020-2023
|
2022-10-26 08:51:08 +08:00
|
|
|
! - Karl D. Hammond <hammondkd@missouri.edu>, University of Missouri, 2022
|
2022-10-24 23:06:32 +08:00
|
|
|
!
|
2022-09-15 10:07:32 +08:00
|
|
|
! The Fortran module tries to follow the API of the C library interface
|
2022-10-24 23:06:32 +08:00
|
|
|
! closely, but like the Python wrapper, it employs an object-oriented
|
2022-09-15 10:07:32 +08:00
|
|
|
! approach. To accommodate the object-oriented approach, all exported
|
|
|
|
! subroutines and functions have to be implemented in Fortran and
|
|
|
|
! call the interfaced C-style functions with adapted calling conventions
|
|
|
|
! as needed. The C library interface functions retain their names
|
|
|
|
! starting with "lammps_", while the Fortran versions start with "lmp_".
|
2020-08-29 08:56:15 +08:00
|
|
|
!
|
|
|
|
MODULE LIBLAMMPS
|
|
|
|
|
2022-10-03 04:28:10 +08:00
|
|
|
USE, INTRINSIC :: ISO_C_BINDING, ONLY: c_ptr, c_null_ptr, C_ASSOCIATED, &
|
|
|
|
C_LOC, c_int, c_int64_t, c_char, c_null_char, c_double, c_size_t, &
|
2022-11-22 12:38:10 +08:00
|
|
|
C_F_POINTER, c_funptr, C_FUNLOC
|
2020-08-29 08:56:15 +08:00
|
|
|
|
|
|
|
IMPLICIT NONE
|
|
|
|
PRIVATE
|
2022-08-12 13:04:25 +08:00
|
|
|
PUBLIC :: lammps, ASSIGNMENT(=)
|
2022-08-10 13:55:17 +08:00
|
|
|
|
|
|
|
! Data type constants for extracting data from global, atom, compute, and fix
|
|
|
|
!
|
|
|
|
! Must be kept in sync with the equivalent declarations in
|
2023-06-11 07:23:11 +08:00
|
|
|
! src/library.h, src/lmptype.h, python/lammps/constants.py, tools/swig/lammps.i,
|
2022-10-29 09:13:50 +08:00
|
|
|
! and examples/COUPLE/plugin/liblammpsplugin.h
|
2022-08-11 06:56:28 +08:00
|
|
|
!
|
2022-09-24 05:23:51 +08:00
|
|
|
! These are NOT part of the API (the part the user sees)
|
2022-10-01 07:50:16 +08:00
|
|
|
INTEGER(c_int), PARAMETER :: &
|
2023-06-11 07:23:11 +08:00
|
|
|
LAMMPS_NONE = -1, & ! no data type assigned (yet)
|
2022-10-24 14:00:48 +08:00
|
|
|
LAMMPS_INT = 0, & ! 32-bit integer (or array)
|
2022-09-26 12:54:18 +08:00
|
|
|
LAMMPS_INT_2D = 1, & ! two-dimensional 32-bit integer array
|
2022-10-24 14:00:48 +08:00
|
|
|
LAMMPS_DOUBLE = 2, & ! 64-bit double (or array)
|
2022-09-26 12:54:18 +08:00
|
|
|
LAMMPS_DOUBLE_2D = 3, & ! two-dimensional 64-bit double array
|
2022-10-24 14:00:48 +08:00
|
|
|
LAMMPS_INT64 = 4, & ! 64-bit integer (or array)
|
2022-09-26 12:54:18 +08:00
|
|
|
LAMMPS_INT64_2D = 5, & ! two-dimensional 64-bit integer array
|
2022-11-30 05:37:15 +08:00
|
|
|
LAMMPS_STRING = 6, & ! string
|
2022-09-26 12:54:18 +08:00
|
|
|
LMP_STYLE_GLOBAL = 0, & ! request global compute/fix/etc. data
|
|
|
|
LMP_STYLE_ATOM = 1, & ! request per-atom compute/fix/etc. data
|
|
|
|
LMP_STYLE_LOCAL = 2, & ! request local compute/fix/etc. data
|
|
|
|
LMP_TYPE_SCALAR = 0, & ! request scalar
|
|
|
|
LMP_TYPE_VECTOR = 1, & ! request vector
|
|
|
|
LMP_TYPE_ARRAY = 2, & ! request array
|
|
|
|
LMP_SIZE_VECTOR = 3, & ! request size of vector
|
|
|
|
LMP_SIZE_ROWS = 4, & ! request rows (actually columns)
|
2022-11-30 05:37:15 +08:00
|
|
|
LMP_SIZE_COLS = 5, & ! request columns (actually rows)
|
2022-09-26 12:54:18 +08:00
|
|
|
LMP_ERROR_WARNING = 0, & ! call Error::warning()
|
|
|
|
LMP_ERROR_ONE = 1, & ! call Error::one() (from this MPI rank)
|
|
|
|
LMP_ERROR_ALL = 2, & ! call Error::all() (from all MPI ranks)
|
|
|
|
LMP_ERROR_WORLD = 4, & ! error on comm->world
|
2022-09-29 11:09:26 +08:00
|
|
|
LMP_ERROR_UNIVERSE = 8, & ! error on comm->universe
|
|
|
|
LMP_VAR_EQUAL = 0, & ! equal-style variables (and compatible)
|
2022-10-01 05:02:26 +08:00
|
|
|
LMP_VAR_ATOM = 1, & ! atom-style variables
|
|
|
|
LMP_VAR_VECTOR = 2, & ! vector variables
|
|
|
|
LMP_VAR_STRING = 3 ! string variables (everything else)
|
2022-09-24 05:23:51 +08:00
|
|
|
|
2022-11-25 11:07:46 +08:00
|
|
|
! Constants we set once (in the constructor) and never need to check again
|
|
|
|
INTEGER(c_int), SAVE :: SIZE_TAGINT, SIZE_BIGINT, SIZE_IMAGEINT
|
|
|
|
|
2022-09-24 05:23:51 +08:00
|
|
|
! "Constants" to use with extract_compute and friends
|
|
|
|
TYPE lammps_style
|
|
|
|
INTEGER(c_int) :: global, atom, local
|
|
|
|
END TYPE lammps_style
|
|
|
|
|
|
|
|
TYPE lammps_type
|
|
|
|
INTEGER(c_int) :: scalar, vector, array
|
|
|
|
END TYPE lammps_type
|
2022-08-10 13:55:17 +08:00
|
|
|
|
2023-06-09 07:12:59 +08:00
|
|
|
TYPE lammps_dtype
|
|
|
|
INTEGER(c_int) :: i32, i64, r64, str
|
|
|
|
END TYPE lammps_dtype
|
|
|
|
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE lammps
|
2022-09-30 08:28:15 +08:00
|
|
|
TYPE(c_ptr) :: handle = c_null_ptr
|
|
|
|
TYPE(lammps_style) :: style
|
|
|
|
TYPE(lammps_type) :: type
|
2023-06-09 07:12:59 +08:00
|
|
|
TYPE(lammps_dtype) :: dtype
|
2022-09-30 08:28:15 +08:00
|
|
|
CONTAINS
|
|
|
|
PROCEDURE :: close => lmp_close
|
|
|
|
PROCEDURE :: error => lmp_error
|
2024-09-20 15:46:36 +08:00
|
|
|
PROCEDURE :: expand => lmp_expand
|
2022-09-30 08:28:15 +08:00
|
|
|
PROCEDURE :: file => lmp_file
|
|
|
|
PROCEDURE :: command => lmp_command
|
|
|
|
PROCEDURE :: commands_list => lmp_commands_list
|
|
|
|
PROCEDURE :: commands_string => lmp_commands_string
|
|
|
|
PROCEDURE :: get_natoms => lmp_get_natoms
|
|
|
|
PROCEDURE :: get_thermo => lmp_get_thermo
|
2023-06-09 07:12:59 +08:00
|
|
|
PROCEDURE :: last_thermo => lmp_last_thermo
|
2022-09-30 08:28:15 +08:00
|
|
|
PROCEDURE :: extract_box => lmp_extract_box
|
|
|
|
PROCEDURE :: reset_box => lmp_reset_box
|
|
|
|
PROCEDURE :: memory_usage => lmp_memory_usage
|
|
|
|
PROCEDURE :: get_mpi_comm => lmp_get_mpi_comm
|
|
|
|
PROCEDURE :: extract_setting => lmp_extract_setting
|
|
|
|
PROCEDURE :: extract_global => lmp_extract_global
|
2024-07-18 19:39:15 +08:00
|
|
|
PROCEDURE :: extract_pair_dimension => lmp_extract_pair_dimension
|
|
|
|
PROCEDURE :: extract_pair => lmp_extract_pair
|
2024-05-25 19:01:28 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_map_atom_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_map_atom_big
|
|
|
|
GENERIC :: map_atom => lmp_map_atom_int, lmp_map_atom_big
|
2022-09-30 08:28:15 +08:00
|
|
|
PROCEDURE :: extract_atom => lmp_extract_atom
|
|
|
|
PROCEDURE :: extract_compute => lmp_extract_compute
|
|
|
|
PROCEDURE :: extract_fix => lmp_extract_fix
|
|
|
|
PROCEDURE :: extract_variable => lmp_extract_variable
|
2022-10-01 07:31:18 +08:00
|
|
|
PROCEDURE :: set_variable => lmp_set_variable
|
2024-01-24 16:51:18 +08:00
|
|
|
PROCEDURE :: set_string_variable => lmp_set_string_variable
|
|
|
|
PROCEDURE :: set_internal_variable => lmp_set_internal_variable
|
2022-10-04 06:44:09 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_atoms_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_atoms_double
|
2022-10-03 04:28:10 +08:00
|
|
|
GENERIC :: gather_atoms => lmp_gather_atoms_int, &
|
|
|
|
lmp_gather_atoms_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_atoms_concat_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_atoms_concat_double
|
|
|
|
GENERIC :: gather_atoms_concat => lmp_gather_atoms_concat_int, &
|
|
|
|
lmp_gather_atoms_concat_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_atoms_subset_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_atoms_subset_double
|
|
|
|
GENERIC :: gather_atoms_subset => lmp_gather_atoms_subset_int, &
|
|
|
|
lmp_gather_atoms_subset_double
|
2022-10-04 06:44:09 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_atoms_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_atoms_double
|
2022-10-03 09:32:42 +08:00
|
|
|
GENERIC :: scatter_atoms => lmp_scatter_atoms_int, &
|
|
|
|
lmp_scatter_atoms_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_atoms_subset_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_atoms_subset_double
|
|
|
|
GENERIC :: scatter_atoms_subset => lmp_scatter_atoms_subset_int, &
|
|
|
|
lmp_scatter_atoms_subset_double
|
2022-10-24 14:00:48 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_bonds_small
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_bonds_big
|
|
|
|
GENERIC :: gather_bonds => lmp_gather_bonds_small, &
|
|
|
|
lmp_gather_bonds_big
|
2023-01-26 18:22:08 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_angles_small
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_angles_big
|
|
|
|
GENERIC :: gather_angles => lmp_gather_angles_small, &
|
|
|
|
lmp_gather_angles_big
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_dihedrals_small
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_dihedrals_big
|
|
|
|
GENERIC :: gather_dihedrals => lmp_gather_dihedrals_small, &
|
|
|
|
lmp_gather_dihedrals_big
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_impropers_small
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_impropers_big
|
|
|
|
GENERIC :: gather_impropers => lmp_gather_impropers_small, &
|
|
|
|
lmp_gather_impropers_big
|
2022-12-03 07:19:42 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_double
|
|
|
|
GENERIC :: gather => lmp_gather_int, lmp_gather_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_concat_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_concat_double
|
|
|
|
GENERIC :: gather_concat => lmp_gather_concat_int, &
|
|
|
|
lmp_gather_concat_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_subset_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_gather_subset_double
|
|
|
|
GENERIC :: gather_subset => lmp_gather_subset_int, &
|
|
|
|
lmp_gather_subset_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_double
|
|
|
|
GENERIC :: scatter => lmp_scatter_int, lmp_scatter_double
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_subset_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_scatter_subset_double
|
|
|
|
GENERIC :: scatter_subset => lmp_scatter_subset_int, &
|
|
|
|
lmp_scatter_subset_double
|
2022-10-19 22:56:54 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_create_atoms_int
|
|
|
|
PROCEDURE, PRIVATE :: lmp_create_atoms_bigbig
|
2022-10-20 13:27:57 +08:00
|
|
|
GENERIC :: create_atoms => lmp_create_atoms_int, &
|
2022-10-19 22:56:54 +08:00
|
|
|
lmp_create_atoms_bigbig
|
2022-10-24 14:00:48 +08:00
|
|
|
PROCEDURE :: find_pair_neighlist => lmp_find_pair_neighlist
|
|
|
|
PROCEDURE :: find_fix_neighlist => lmp_find_fix_neighlist
|
|
|
|
PROCEDURE :: find_compute_neighlist => lmp_find_compute_neighlist
|
|
|
|
PROCEDURE :: neighlist_num_elements => lmp_neighlist_num_elements
|
2022-10-31 05:42:10 +08:00
|
|
|
PROCEDURE :: neighlist_element_neighbors => lmp_neighlist_element_neighbors
|
2022-09-30 08:28:15 +08:00
|
|
|
PROCEDURE :: version => lmp_version
|
2022-10-20 07:40:12 +08:00
|
|
|
PROCEDURE, NOPASS :: get_os_info => lmp_get_os_info
|
|
|
|
PROCEDURE, NOPASS :: config_has_mpi_support => lmp_config_has_mpi_support
|
|
|
|
PROCEDURE, NOPASS :: config_has_gzip_support => lmp_config_has_gzip_support
|
|
|
|
PROCEDURE, NOPASS :: config_has_png_support => lmp_config_has_png_support
|
|
|
|
PROCEDURE, NOPASS :: config_has_jpeg_support => lmp_config_has_jpeg_support
|
|
|
|
PROCEDURE, NOPASS :: config_has_ffmpeg_support &
|
|
|
|
=> lmp_config_has_ffmpeg_support
|
2024-08-07 22:37:20 +08:00
|
|
|
PROCEDURE, NOPASS :: config_has_curl_support => lmp_config_has_curl_support
|
2022-10-20 07:40:12 +08:00
|
|
|
PROCEDURE, NOPASS :: config_has_exceptions => lmp_config_has_exceptions
|
|
|
|
PROCEDURE, NOPASS :: config_has_package => lmp_config_has_package
|
|
|
|
PROCEDURE, NOPASS :: config_package_count => lammps_config_package_count
|
2022-10-26 08:51:08 +08:00
|
|
|
PROCEDURE :: config_package_name => lmp_config_package_name
|
|
|
|
PROCEDURE :: installed_packages => lmp_installed_packages
|
2022-10-21 09:24:59 +08:00
|
|
|
PROCEDURE, NOPASS :: config_accelerator => lmp_config_accelerator
|
|
|
|
PROCEDURE, NOPASS :: has_gpu_device => lmp_has_gpu_device
|
|
|
|
PROCEDURE, NOPASS :: get_gpu_device_info => lmp_get_gpu_device_info
|
|
|
|
PROCEDURE :: has_style => lmp_has_style
|
|
|
|
PROCEDURE :: style_count => lmp_style_count
|
|
|
|
PROCEDURE :: style_name => lmp_style_name
|
2022-10-26 08:51:08 +08:00
|
|
|
PROCEDURE :: has_id => lmp_has_id
|
|
|
|
PROCEDURE :: id_count => lmp_id_count
|
|
|
|
PROCEDURE :: id_name => lmp_id_name
|
2022-10-26 13:02:54 +08:00
|
|
|
PROCEDURE, NOPASS :: plugin_count => lammps_plugin_count
|
|
|
|
PROCEDURE :: plugin_name => lmp_plugin_name
|
2022-10-20 07:40:12 +08:00
|
|
|
PROCEDURE :: encode_image_flags => lmp_encode_image_flags
|
2022-10-20 13:27:57 +08:00
|
|
|
PROCEDURE, PRIVATE :: lmp_decode_image_flags
|
|
|
|
PROCEDURE, PRIVATE :: lmp_decode_image_flags_bigbig
|
|
|
|
GENERIC :: decode_image_flags => lmp_decode_image_flags, &
|
|
|
|
lmp_decode_image_flags_bigbig
|
2022-11-22 12:38:10 +08:00
|
|
|
PROCEDURE :: set_fix_external_callback => lmp_set_fix_external_callback
|
2022-11-30 13:23:14 +08:00
|
|
|
PROCEDURE :: fix_external_get_force => lmp_fix_external_get_force
|
|
|
|
PROCEDURE :: fix_external_set_energy_global &
|
|
|
|
=> lmp_fix_external_set_energy_global
|
|
|
|
PROCEDURE :: fix_external_set_virial_global &
|
|
|
|
=> lmp_fix_external_set_virial_global
|
|
|
|
PROCEDURE :: fix_external_set_energy_peratom &
|
|
|
|
=> lmp_fix_external_set_energy_peratom
|
|
|
|
PROCEDURE :: fix_external_set_virial_peratom &
|
|
|
|
=> lmp_fix_external_set_virial_peratom
|
|
|
|
PROCEDURE :: fix_external_set_vector_length &
|
|
|
|
=> lmp_fix_external_set_vector_length
|
|
|
|
PROCEDURE :: fix_external_set_vector => lmp_fix_external_set_vector
|
2022-09-30 08:28:15 +08:00
|
|
|
PROCEDURE :: flush_buffers => lmp_flush_buffers
|
|
|
|
PROCEDURE :: is_running => lmp_is_running
|
|
|
|
PROCEDURE :: force_timeout => lmp_force_timeout
|
|
|
|
PROCEDURE :: has_error => lmp_has_error
|
|
|
|
PROCEDURE :: get_last_error_message => lmp_get_last_error_message
|
2020-08-29 08:56:15 +08:00
|
|
|
END TYPE lammps
|
|
|
|
|
|
|
|
INTERFACE lammps
|
2022-08-13 00:33:28 +08:00
|
|
|
MODULE PROCEDURE lmp_open
|
2020-08-29 08:56:15 +08:00
|
|
|
END INTERFACE lammps
|
|
|
|
|
2022-08-12 13:04:25 +08:00
|
|
|
! Constants to use in working with lammps_data
|
|
|
|
ENUM, BIND(C)
|
|
|
|
ENUMERATOR :: DATA_INT, DATA_INT_1D, DATA_INT_2D
|
|
|
|
ENUMERATOR :: DATA_INT64, DATA_INT64_1D, DATA_INT64_2D
|
|
|
|
ENUMERATOR :: DATA_DOUBLE, DATA_DOUBLE_1D, DATA_DOUBLE_2D
|
|
|
|
ENUMERATOR :: DATA_STRING
|
|
|
|
END ENUM
|
2022-08-13 00:33:28 +08:00
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
! Base class for receiving LAMMPS data (to reduce code duplication)
|
|
|
|
TYPE lammps_data_baseclass
|
|
|
|
INTEGER(c_int) :: datatype = -1_c_int
|
|
|
|
! in case we need to call the Error class in an assignment
|
|
|
|
CLASS(lammps), POINTER, PRIVATE :: lammps_instance => NULL()
|
|
|
|
END TYPE lammps_data_baseclass
|
|
|
|
|
2022-08-12 13:04:25 +08:00
|
|
|
! Derived type for receiving LAMMPS data (in lieu of the ability to type cast
|
2023-06-09 07:12:59 +08:00
|
|
|
! pointers). Used for extract_compute, extract_atom, last_thermo
|
2022-09-29 11:09:26 +08:00
|
|
|
TYPE, EXTENDS(lammps_data_baseclass) :: lammps_data
|
2022-09-26 12:54:18 +08:00
|
|
|
INTEGER(c_int), POINTER :: i32 => NULL()
|
|
|
|
INTEGER(c_int), DIMENSION(:), POINTER :: i32_vec => NULL()
|
|
|
|
INTEGER(c_int64_t), POINTER :: i64 => NULL()
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), POINTER :: i64_vec => NULL()
|
|
|
|
REAL(c_double), POINTER :: r64 => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:), POINTER :: r64_vec => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:,:), POINTER :: r64_mat => NULL()
|
2022-08-12 13:04:25 +08:00
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: str
|
|
|
|
END TYPE lammps_data
|
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
! Derived type for holding LAMMPS fix data
|
|
|
|
! Done this way because fix global data are not pointers, but computed
|
|
|
|
! on-the-fly, whereas per-atom and local data are pointers to the actual
|
|
|
|
! array. Doing it this way saves the user from having to explicitly
|
|
|
|
! deallocate all of the pointers.
|
|
|
|
TYPE, EXTENDS(lammps_data_baseclass) :: lammps_fix_data
|
|
|
|
REAL(c_double) :: r64
|
|
|
|
REAL(c_double), DIMENSION(:), POINTER :: r64_vec => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:,:), POINTER :: r64_mat => NULL()
|
|
|
|
END TYPE lammps_fix_data
|
|
|
|
|
|
|
|
! Derived type for holding LAMMPS variable data
|
|
|
|
! Done this way because extract_variable calculates variable values, it does
|
|
|
|
! not return pointers to LAMMPS data.
|
|
|
|
TYPE, EXTENDS(lammps_data_baseclass) :: lammps_variable_data
|
|
|
|
REAL(c_double) :: r64
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE :: r64_vec
|
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: str
|
|
|
|
END TYPE lammps_variable_data
|
|
|
|
|
2022-10-20 07:40:12 +08:00
|
|
|
TYPE, EXTENDS(lammps_data_baseclass) :: lammps_image_data
|
|
|
|
INTEGER(c_int) :: i32
|
|
|
|
INTEGER(c_int64_t) :: i64
|
|
|
|
END TYPE lammps_image_data
|
|
|
|
|
2022-08-13 00:33:28 +08:00
|
|
|
! This overloads the assignment operator (=) so that assignments of the
|
|
|
|
! form
|
|
|
|
! nlocal = extract_global('nlocal')
|
|
|
|
! which are of the form "pointer to double = type(lammps_data)" result in
|
|
|
|
! re-associating the pointer on the left with the appropriate piece of
|
2022-09-24 05:23:51 +08:00
|
|
|
! LAMMPS data (after checking type-kind-rank compatibility)
|
2022-08-12 13:04:25 +08:00
|
|
|
INTERFACE ASSIGNMENT(=)
|
|
|
|
MODULE PROCEDURE assign_int_to_lammps_data, assign_int64_to_lammps_data, &
|
2022-09-23 08:16:15 +08:00
|
|
|
assign_intvec_to_lammps_data, assign_int64vec_to_lammps_data, &
|
2022-08-12 13:04:25 +08:00
|
|
|
assign_double_to_lammps_data, assign_doublevec_to_lammps_data, &
|
2022-09-23 08:16:15 +08:00
|
|
|
assign_doublemat_to_lammps_data, &
|
2022-08-12 13:04:25 +08:00
|
|
|
assign_string_to_lammps_data
|
2022-09-29 11:09:26 +08:00
|
|
|
! We handle fix data (slightly) differently
|
|
|
|
MODULE PROCEDURE assign_double_to_lammps_fix_data, &
|
|
|
|
assign_doublevec_to_lammps_fix_data, &
|
|
|
|
assign_doublemat_to_lammps_fix_data
|
2022-09-30 08:28:15 +08:00
|
|
|
! Variables, too
|
|
|
|
MODULE PROCEDURE assign_double_to_lammps_variable_data, &
|
2022-09-30 12:39:39 +08:00
|
|
|
assign_doublevec_to_lammps_variable_data, &
|
2022-09-30 08:28:15 +08:00
|
|
|
assign_string_to_lammps_variable_data
|
2022-10-20 07:40:12 +08:00
|
|
|
! Image data, too
|
|
|
|
MODULE PROCEDURE assign_int_to_lammps_image_data, &
|
|
|
|
assign_int64_to_lammps_image_data
|
2022-08-12 13:04:25 +08:00
|
|
|
END INTERFACE
|
|
|
|
|
2022-11-22 12:38:10 +08:00
|
|
|
! Interface templates for fix external callbacks
|
|
|
|
ABSTRACT INTERFACE
|
|
|
|
SUBROUTINE external_callback_smallsmall(caller, timestep, ids, x, fexternal)
|
|
|
|
IMPORT :: c_int, c_double
|
2022-11-30 05:37:15 +08:00
|
|
|
CLASS(*), INTENT(INOUT) :: caller
|
2022-11-22 12:38:10 +08:00
|
|
|
INTEGER(c_int), INTENT(IN) :: timestep
|
|
|
|
INTEGER(c_int), DIMENSION(:), INTENT(IN) :: ids
|
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(IN) :: x
|
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(OUT) :: fexternal
|
|
|
|
END SUBROUTINE external_callback_smallsmall
|
|
|
|
SUBROUTINE external_callback_smallbig(caller, timestep, ids, x, fexternal)
|
|
|
|
IMPORT :: c_int, c_double, c_int64_t
|
2022-11-30 05:37:15 +08:00
|
|
|
CLASS(*), INTENT(INOUT) :: caller
|
2022-11-22 12:38:10 +08:00
|
|
|
INTEGER(c_int64_t), INTENT(IN) :: timestep
|
|
|
|
INTEGER(c_int), DIMENSION(:), INTENT(IN) :: ids
|
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(IN) :: x
|
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(OUT) :: fexternal
|
|
|
|
END SUBROUTINE external_callback_smallbig
|
|
|
|
SUBROUTINE external_callback_bigbig(caller, timestep, ids, x, fexternal)
|
|
|
|
IMPORT :: c_double, c_int64_t
|
2022-11-30 05:37:15 +08:00
|
|
|
CLASS(*), INTENT(INOUT) :: caller
|
2022-11-22 12:38:10 +08:00
|
|
|
INTEGER(c_int64_t), INTENT(IN) :: timestep
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), INTENT(IN) :: ids
|
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(IN) :: x
|
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(OUT) :: fexternal
|
|
|
|
END SUBROUTINE external_callback_bigbig
|
|
|
|
END INTERFACE
|
|
|
|
|
|
|
|
! Derived type for fix external callback data
|
|
|
|
TYPE fix_external_data
|
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: id
|
|
|
|
PROCEDURE(external_callback_smallsmall), NOPASS, POINTER :: &
|
|
|
|
callback_smallsmall => NULL()
|
|
|
|
PROCEDURE(external_callback_smallbig), NOPASS, POINTER :: &
|
|
|
|
callback_smallbig => NULL()
|
|
|
|
PROCEDURE(external_callback_bigbig), NOPASS, POINTER :: &
|
|
|
|
callback_bigbig => NULL()
|
|
|
|
CLASS(*), POINTER :: caller => NULL()
|
|
|
|
CLASS(lammps), POINTER :: lammps_instance => NULL()
|
|
|
|
END TYPE fix_external_data
|
2022-12-23 12:09:20 +08:00
|
|
|
! constructor to make Fortran 2003 compilers happy (F2008 should be OK)
|
|
|
|
INTERFACE fix_external_data
|
|
|
|
MODULE PROCEDURE construct_fix_external_data
|
|
|
|
END INTERFACE fix_external_data
|
2022-11-22 12:38:10 +08:00
|
|
|
|
|
|
|
! Array used to store Fortran-facing callback functions for fix external
|
|
|
|
TYPE(fix_external_data), DIMENSION(:), ALLOCATABLE, TARGET, SAVE :: ext_data
|
|
|
|
|
2020-08-29 08:56:15 +08:00
|
|
|
! interface definitions for calling functions in library.cpp
|
|
|
|
INTERFACE
|
2022-09-15 10:07:32 +08:00
|
|
|
FUNCTION lammps_open(argc, argv, comm) BIND(C,name='lammps_open_fortran')
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int), VALUE, INTENT(IN) :: argc, comm
|
|
|
|
TYPE(c_ptr), DIMENSION(*), INTENT(IN) :: argv
|
|
|
|
TYPE(c_ptr) :: lammps_open
|
|
|
|
END FUNCTION lammps_open
|
|
|
|
|
|
|
|
FUNCTION lammps_open_no_mpi(argc, argv, handle) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int), VALUE, INTENT(IN) :: argc
|
|
|
|
TYPE(c_ptr), DIMENSION(*), INTENT(IN) :: argv
|
|
|
|
TYPE(c_ptr), VALUE, INTENT(IN) :: handle
|
|
|
|
TYPE(c_ptr) :: lammps_open_no_mpi
|
|
|
|
END FUNCTION lammps_open_no_mpi
|
|
|
|
|
|
|
|
SUBROUTINE lammps_close(handle) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
END SUBROUTINE lammps_close
|
|
|
|
|
|
|
|
SUBROUTINE lammps_mpi_init() BIND(C)
|
|
|
|
END SUBROUTINE lammps_mpi_init
|
|
|
|
|
|
|
|
SUBROUTINE lammps_mpi_finalize() BIND(C)
|
|
|
|
END SUBROUTINE lammps_mpi_finalize
|
|
|
|
|
|
|
|
SUBROUTINE lammps_kokkos_finalize() BIND(C)
|
|
|
|
END SUBROUTINE lammps_kokkos_finalize
|
|
|
|
|
2022-09-24 04:28:15 +08:00
|
|
|
SUBROUTINE lammps_error(handle, error_type, error_text) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
INTEGER(c_int), VALUE :: error_type
|
|
|
|
TYPE(c_ptr), VALUE :: error_text
|
|
|
|
END SUBROUTINE lammps_error
|
|
|
|
|
2024-09-20 15:46:36 +08:00
|
|
|
FUNCTION lammps_expand(handle, line) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: line
|
|
|
|
TYPE(c_ptr) :: lammps_expand
|
|
|
|
END FUNCTION lammps_expand
|
|
|
|
|
2022-09-15 10:07:32 +08:00
|
|
|
SUBROUTINE lammps_file(handle, filename) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
TYPE(c_ptr), VALUE :: filename
|
|
|
|
END SUBROUTINE lammps_file
|
|
|
|
|
|
|
|
SUBROUTINE lammps_command(handle, cmd) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: cmd
|
2022-09-15 10:07:32 +08:00
|
|
|
END SUBROUTINE lammps_command
|
|
|
|
|
|
|
|
SUBROUTINE lammps_commands_list(handle, ncmd, cmds) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: ncmd
|
2022-09-15 10:07:32 +08:00
|
|
|
TYPE(c_ptr), DIMENSION(*), INTENT(IN) :: cmds
|
|
|
|
END SUBROUTINE lammps_commands_list
|
|
|
|
|
|
|
|
SUBROUTINE lammps_commands_string(handle, str) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: str
|
2022-09-15 10:07:32 +08:00
|
|
|
END SUBROUTINE lammps_commands_string
|
|
|
|
|
|
|
|
FUNCTION lammps_get_natoms(handle) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_double
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
2022-09-15 10:07:32 +08:00
|
|
|
REAL(c_double) :: lammps_get_natoms
|
|
|
|
END FUNCTION lammps_get_natoms
|
|
|
|
|
|
|
|
FUNCTION lammps_get_thermo(handle,name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_double
|
|
|
|
IMPLICIT NONE
|
|
|
|
REAL(c_double) :: lammps_get_thermo
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: name
|
2022-09-15 10:07:32 +08:00
|
|
|
END FUNCTION lammps_get_thermo
|
|
|
|
|
2023-06-09 07:12:59 +08:00
|
|
|
FUNCTION lammps_last_thermo(handle,what,index) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr) :: lammps_last_thermo
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: what
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: index
|
|
|
|
END FUNCTION lammps_last_thermo
|
|
|
|
|
2022-09-15 10:07:32 +08:00
|
|
|
SUBROUTINE lammps_extract_box(handle,boxlo,boxhi,xy,yz,xz,pflags, &
|
|
|
|
boxflag) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_double, c_int
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, boxlo, boxhi, xy, yz, xz, &
|
|
|
|
pflags, boxflag
|
2022-09-15 10:07:32 +08:00
|
|
|
END SUBROUTINE lammps_extract_box
|
|
|
|
|
|
|
|
SUBROUTINE lammps_reset_box(handle,boxlo,boxhi,xy,yz,xz) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_double
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
REAL(c_double), DIMENSION(3), INTENT(IN) :: boxlo, boxhi
|
|
|
|
REAL(c_double), INTENT(IN), VALUE :: xy, yz, xz
|
2022-09-15 10:07:32 +08:00
|
|
|
END SUBROUTINE lammps_reset_box
|
|
|
|
|
|
|
|
SUBROUTINE lammps_memory_usage(handle,meminfo) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_double
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
|
|
|
REAL(c_double), DIMENSION(*), INTENT(OUT) :: meminfo
|
2022-09-15 10:07:32 +08:00
|
|
|
END SUBROUTINE lammps_memory_usage
|
|
|
|
|
|
|
|
FUNCTION lammps_get_mpi_comm(handle) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle
|
2022-09-15 10:07:32 +08:00
|
|
|
INTEGER(c_int) :: lammps_get_mpi_comm
|
|
|
|
END FUNCTION lammps_get_mpi_comm
|
|
|
|
|
|
|
|
FUNCTION lammps_extract_setting(handle,keyword) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, keyword
|
2022-09-15 10:07:32 +08:00
|
|
|
INTEGER(c_int) :: lammps_extract_setting
|
|
|
|
END FUNCTION lammps_extract_setting
|
|
|
|
|
|
|
|
FUNCTION lammps_extract_global_datatype(handle,name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
2022-09-15 10:07:32 +08:00
|
|
|
INTEGER(c_int) :: lammps_extract_global_datatype
|
|
|
|
END FUNCTION lammps_extract_global_datatype
|
|
|
|
|
|
|
|
FUNCTION lammps_extract_global(handle, name) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
2022-09-15 10:07:32 +08:00
|
|
|
TYPE(c_ptr) :: lammps_extract_global
|
|
|
|
END FUNCTION lammps_extract_global
|
|
|
|
|
2024-07-18 19:39:15 +08:00
|
|
|
FUNCTION lammps_extract_pair_dimension(handle,name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
|
|
|
INTEGER(c_int) :: lammps_extract_pair_dimension
|
|
|
|
END FUNCTION lammps_extract_pair_dimension
|
|
|
|
|
|
|
|
FUNCTION lammps_extract_pair(handle, name) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
|
|
|
TYPE(c_ptr) :: lammps_extract_pair
|
|
|
|
END FUNCTION lammps_extract_pair
|
|
|
|
|
2024-05-25 19:01:28 +08:00
|
|
|
FUNCTION lammps_map_atom(handle, tag) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, tag
|
|
|
|
INTEGER(c_int) :: lammps_map_atom
|
|
|
|
END FUNCTION lammps_map_atom
|
|
|
|
|
2022-09-23 08:16:15 +08:00
|
|
|
FUNCTION lammps_extract_atom_datatype(handle, name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
2022-09-23 08:16:15 +08:00
|
|
|
INTEGER(c_int) :: lammps_extract_atom_datatype
|
|
|
|
END FUNCTION lammps_extract_atom_datatype
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2024-08-31 10:50:42 +08:00
|
|
|
FUNCTION lammps_extract_atom_size(handle, name, dtype) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: dtype
|
|
|
|
INTEGER(c_int) :: lammps_extract_atom_size
|
|
|
|
END FUNCTION lammps_extract_atom_size
|
|
|
|
|
2022-09-23 08:16:15 +08:00
|
|
|
FUNCTION lammps_extract_atom(handle, name) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
2022-09-24 05:23:51 +08:00
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
2022-09-23 08:16:15 +08:00
|
|
|
TYPE(c_ptr) :: lammps_extract_atom
|
|
|
|
END FUNCTION lammps_extract_atom
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2022-09-24 05:23:51 +08:00
|
|
|
FUNCTION lammps_extract_compute(handle, id, style, type) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, id
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: style, type
|
|
|
|
TYPE(c_ptr) :: lammps_extract_compute
|
|
|
|
END FUNCTION lammps_extract_compute
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
FUNCTION lammps_extract_fix(handle, id, style, type, nrow, ncol) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, id
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: style, type, nrow, ncol
|
|
|
|
TYPE(c_ptr) :: lammps_extract_fix
|
|
|
|
END FUNCTION lammps_extract_fix
|
|
|
|
|
|
|
|
FUNCTION lammps_extract_variable_datatype(handle,name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name
|
|
|
|
INTEGER(c_int) :: lammps_extract_variable_datatype
|
|
|
|
END FUNCTION lammps_extract_variable_datatype
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
FUNCTION lammps_extract_variable(handle, name, group) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: handle, name, group
|
|
|
|
TYPE(c_ptr) :: lammps_extract_variable
|
|
|
|
END FUNCTION lammps_extract_variable
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lammps_set_variable(handle, name, str) BIND(C)
|
2022-10-01 07:31:18 +08:00
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
2022-10-01 07:50:16 +08:00
|
|
|
TYPE(c_ptr), VALUE :: handle, name, str
|
|
|
|
INTEGER(c_int) :: lammps_set_variable
|
2022-10-01 07:31:18 +08:00
|
|
|
END FUNCTION lammps_set_variable
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2024-01-24 16:51:18 +08:00
|
|
|
FUNCTION lammps_set_string_variable(handle, name, str) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, str
|
|
|
|
INTEGER(c_int) :: lammps_set_string_variable
|
|
|
|
END FUNCTION lammps_set_string_variable
|
|
|
|
|
|
|
|
FUNCTION lammps_set_internal_variable(handle, name, val) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr, c_double
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name
|
|
|
|
REAL(c_double), VALUE :: val
|
|
|
|
INTEGER(c_int) :: lammps_set_internal_variable
|
|
|
|
END FUNCTION lammps_set_internal_variable
|
|
|
|
|
2022-10-03 04:28:10 +08:00
|
|
|
SUBROUTINE lammps_gather_atoms(handle, name, type, count, data) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count
|
|
|
|
END SUBROUTINE lammps_gather_atoms
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2022-10-03 04:28:10 +08:00
|
|
|
SUBROUTINE lammps_gather_atoms_concat(handle, name, type, count, data) &
|
|
|
|
BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count
|
|
|
|
END SUBROUTINE lammps_gather_atoms_concat
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-03 04:28:10 +08:00
|
|
|
SUBROUTINE lammps_gather_atoms_subset(handle, name, type, count, ndata, &
|
|
|
|
ids, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, ids, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count, ndata
|
|
|
|
END SUBROUTINE lammps_gather_atoms_subset
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-03 09:32:42 +08:00
|
|
|
SUBROUTINE lammps_scatter_atoms(handle, name, type, count, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count
|
|
|
|
END SUBROUTINE lammps_scatter_atoms
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-03 09:32:42 +08:00
|
|
|
SUBROUTINE lammps_scatter_atoms_subset(handle, name, type, count, &
|
|
|
|
ndata, ids, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, ids, data
|
|
|
|
INTEGER(c_int), VALUE :: count, ndata, type
|
|
|
|
END SUBROUTINE lammps_scatter_atoms_subset
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
SUBROUTINE lammps_gather_bonds(handle, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, data
|
|
|
|
END SUBROUTINE lammps_gather_bonds
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2023-01-26 18:22:08 +08:00
|
|
|
SUBROUTINE lammps_gather_angles(handle, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, data
|
|
|
|
END SUBROUTINE lammps_gather_angles
|
|
|
|
|
|
|
|
SUBROUTINE lammps_gather_dihedrals(handle, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, data
|
|
|
|
END SUBROUTINE lammps_gather_dihedrals
|
|
|
|
|
|
|
|
SUBROUTINE lammps_gather_impropers(handle, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, data
|
|
|
|
END SUBROUTINE lammps_gather_impropers
|
|
|
|
|
2022-12-03 07:19:42 +08:00
|
|
|
SUBROUTINE lammps_gather(handle, name, type, count, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count
|
|
|
|
END SUBROUTINE lammps_gather
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-12-03 07:19:42 +08:00
|
|
|
SUBROUTINE lammps_gather_concat(handle, name, type, count, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count
|
|
|
|
END SUBROUTINE lammps_gather_concat
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-12-03 07:19:42 +08:00
|
|
|
SUBROUTINE lammps_gather_subset(handle, name, type, count, ndata, ids, &
|
|
|
|
data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, ids, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count, ndata
|
|
|
|
END SUBROUTINE lammps_gather_subset
|
|
|
|
|
|
|
|
SUBROUTINE lammps_scatter(handle, name, type, count, data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, data
|
|
|
|
INTEGER(c_int), VALUE :: type, count
|
|
|
|
END SUBROUTINE lammps_scatter
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-12-03 07:19:42 +08:00
|
|
|
SUBROUTINE lammps_scatter_subset(handle, name, type, count, ndata, ids, &
|
|
|
|
data) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, name, ids, data
|
|
|
|
INTEGER(c_int), VALUE :: count, ndata, type
|
|
|
|
END SUBROUTINE lammps_scatter_subset
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
FUNCTION lammps_create_atoms(handle, n, id, type, x, v, image, bexpand) &
|
|
|
|
BIND(C)
|
2022-10-19 22:56:54 +08:00
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id, type, x, v, image
|
|
|
|
INTEGER(c_int), VALUE :: n, bexpand
|
2022-10-24 14:00:48 +08:00
|
|
|
INTEGER(c_int) :: lammps_create_atoms
|
2022-10-19 22:56:54 +08:00
|
|
|
END FUNCTION lammps_create_atoms
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
FUNCTION lammps_find_pair_neighlist(handle, style, exact, nsub, reqid) &
|
|
|
|
BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, style
|
|
|
|
INTEGER(c_int), VALUE :: exact, nsub, reqid
|
|
|
|
INTEGER(c_int) :: lammps_find_pair_neighlist
|
|
|
|
END FUNCTION lammps_find_pair_neighlist
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
FUNCTION lammps_find_fix_neighlist(handle, id, reqid) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id
|
|
|
|
INTEGER(c_int), VALUE :: reqid
|
|
|
|
INTEGER(c_int) :: lammps_find_fix_neighlist
|
|
|
|
END FUNCTION lammps_find_fix_neighlist
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
FUNCTION lammps_find_compute_neighlist(handle, id, reqid) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id
|
|
|
|
INTEGER(c_int), VALUE :: reqid
|
|
|
|
INTEGER(c_int) :: lammps_find_compute_neighlist
|
|
|
|
END FUNCTION lammps_find_compute_neighlist
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
FUNCTION lammps_neighlist_num_elements(handle, idx) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
INTEGER(c_int), VALUE :: idx
|
|
|
|
INTEGER(c_int) :: lammps_neighlist_num_elements
|
|
|
|
END FUNCTION lammps_neighlist_num_elements
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
SUBROUTINE lammps_neighlist_element_neighbors(handle, idx, element, &
|
|
|
|
iatom, numneigh, neighbors) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
INTEGER(c_int), VALUE :: idx, element
|
|
|
|
INTEGER(c_int) :: iatom, numneigh
|
|
|
|
TYPE(c_ptr) :: neighbors
|
|
|
|
END SUBROUTINE lammps_neighlist_element_neighbors
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-09-15 10:07:32 +08:00
|
|
|
FUNCTION lammps_version(handle) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
INTEGER(c_int) :: lammps_version
|
|
|
|
END FUNCTION lammps_version
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE lammps_get_os_info(buffer, buf_size) BIND(C)
|
2022-10-04 11:49:17 +08:00
|
|
|
IMPORT :: c_ptr, c_int
|
2022-09-30 08:28:15 +08:00
|
|
|
IMPLICIT NONE
|
2022-10-04 11:49:17 +08:00
|
|
|
TYPE(c_ptr), VALUE :: buffer
|
|
|
|
INTEGER(c_int), VALUE :: buf_size
|
2022-09-30 08:28:15 +08:00
|
|
|
END SUBROUTINE lammps_get_os_info
|
|
|
|
|
|
|
|
FUNCTION lammps_config_has_mpi_support() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_config_has_mpi_support
|
|
|
|
END FUNCTION lammps_config_has_mpi_support
|
|
|
|
|
|
|
|
FUNCTION lammps_config_has_gzip_support() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_config_has_gzip_support
|
|
|
|
END FUNCTION lammps_config_has_gzip_support
|
|
|
|
|
|
|
|
FUNCTION lammps_config_has_png_support() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_config_has_png_support
|
|
|
|
END FUNCTION lammps_config_has_png_support
|
|
|
|
|
|
|
|
FUNCTION lammps_config_has_jpeg_support() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_config_has_jpeg_support
|
|
|
|
END FUNCTION lammps_config_has_jpeg_support
|
|
|
|
|
|
|
|
FUNCTION lammps_config_has_ffmpeg_support() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_config_has_ffmpeg_support
|
|
|
|
END FUNCTION lammps_config_has_ffmpeg_support
|
|
|
|
|
2024-08-07 22:37:20 +08:00
|
|
|
FUNCTION lammps_config_has_curl_support() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_config_has_curl_support
|
|
|
|
END FUNCTION lammps_config_has_curl_support
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
FUNCTION lammps_config_has_exceptions() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
2022-10-01 07:50:16 +08:00
|
|
|
INTEGER(c_int) :: lammps_config_has_exceptions
|
2022-09-30 08:28:15 +08:00
|
|
|
END FUNCTION lammps_config_has_exceptions
|
|
|
|
|
|
|
|
FUNCTION lammps_config_has_package(name) BIND(C)
|
2022-10-04 11:49:17 +08:00
|
|
|
IMPORT :: c_int, c_ptr
|
2022-09-30 08:28:15 +08:00
|
|
|
IMPLICIT NONE
|
2022-10-04 11:49:17 +08:00
|
|
|
TYPE(c_ptr), VALUE :: name
|
2022-10-01 07:50:16 +08:00
|
|
|
INTEGER(c_int) :: lammps_config_has_package
|
2022-09-30 08:28:15 +08:00
|
|
|
END FUNCTION lammps_config_has_package
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
FUNCTION lammps_config_package_count() BIND(C)
|
2022-10-04 11:49:17 +08:00
|
|
|
IMPORT :: c_int
|
2022-09-30 08:28:15 +08:00
|
|
|
IMPLICIT NONE
|
2022-10-04 11:49:17 +08:00
|
|
|
INTEGER(c_int) :: lammps_config_package_count
|
2022-09-30 08:28:15 +08:00
|
|
|
END FUNCTION lammps_config_package_count
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lammps_config_package_name(idx, buffer, buf_size) BIND(C)
|
2022-10-04 11:49:17 +08:00
|
|
|
IMPORT :: c_int, c_ptr
|
2022-09-30 08:28:15 +08:00
|
|
|
IMPLICIT NONE
|
2022-10-04 11:49:17 +08:00
|
|
|
INTEGER(c_int) :: lammps_config_package_name
|
|
|
|
INTEGER(c_int), VALUE :: idx, buf_size
|
|
|
|
TYPE(c_ptr), VALUE :: buffer
|
2022-09-30 08:28:15 +08:00
|
|
|
END FUNCTION lammps_config_package_name
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-21 09:24:59 +08:00
|
|
|
FUNCTION lammps_config_accelerator(package, category, setting) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: package, category, setting
|
|
|
|
INTEGER(c_int) :: lammps_config_accelerator
|
|
|
|
END FUNCTION lammps_config_accelerator
|
|
|
|
|
|
|
|
FUNCTION lammps_has_gpu_device() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_has_gpu_device
|
|
|
|
END FUNCTION lammps_has_gpu_device
|
|
|
|
|
|
|
|
SUBROUTINE lammps_get_gpu_device_info(buffer, buf_size) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: buffer
|
|
|
|
INTEGER(c_int), VALUE :: buf_size
|
|
|
|
END SUBROUTINE lammps_get_gpu_device_info
|
|
|
|
|
|
|
|
FUNCTION lammps_has_style(handle, category, name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, category, name
|
|
|
|
INTEGER(c_int) :: lammps_has_style
|
|
|
|
END FUNCTION lammps_has_style
|
|
|
|
|
|
|
|
FUNCTION lammps_style_count(handle, category) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, category
|
|
|
|
INTEGER(c_int) :: lammps_style_count
|
|
|
|
END FUNCTION lammps_style_count
|
|
|
|
|
|
|
|
FUNCTION lammps_style_name(handle, category, idx, buffer, buf_size) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, category, buffer
|
|
|
|
INTEGER(c_int), VALUE :: idx, buf_size
|
|
|
|
INTEGER(c_int) :: lammps_style_name
|
|
|
|
END FUNCTION lammps_style_name
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-26 08:51:08 +08:00
|
|
|
FUNCTION lammps_has_id(handle, category, name) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, category, name
|
|
|
|
INTEGER(c_int) :: lammps_has_id
|
|
|
|
END FUNCTION lammps_has_id
|
|
|
|
|
|
|
|
FUNCTION lammps_id_count(handle, category) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, category
|
|
|
|
INTEGER(c_int) :: lammps_id_count
|
|
|
|
END FUNCTION lammps_id_count
|
|
|
|
|
|
|
|
FUNCTION lammps_id_name(handle, category, idx, buffer, buf_size) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, category, buffer
|
|
|
|
INTEGER(c_int), VALUE :: idx, buf_size
|
|
|
|
INTEGER(c_int) :: lammps_id_name
|
|
|
|
END FUNCTION lammps_id_name
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-26 13:02:54 +08:00
|
|
|
FUNCTION lammps_plugin_count() BIND(C)
|
|
|
|
IMPORT :: c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int) :: lammps_plugin_count
|
|
|
|
END FUNCTION lammps_plugin_count
|
|
|
|
|
|
|
|
FUNCTION lammps_plugin_name(idx, stylebuf, namebuf, buf_size) BIND(C)
|
|
|
|
IMPORT :: c_int, c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_int), VALUE :: idx, buf_size
|
|
|
|
TYPE(c_ptr), VALUE :: stylebuf, namebuf
|
|
|
|
INTEGER(c_int) :: lammps_plugin_name
|
|
|
|
END FUNCTION lammps_plugin_name
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-20 07:40:12 +08:00
|
|
|
! We don't call lammps_encode_image_flags because its interface is
|
2022-10-20 13:27:57 +08:00
|
|
|
! ambiguous: we don't know sizeof(imageint) prior to compile time.
|
2022-10-20 07:40:12 +08:00
|
|
|
! It is re-written in Fortran below. It was easier to do the same for
|
|
|
|
! lammps_decode_image_flags's equivalent.
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-11-22 12:38:10 +08:00
|
|
|
SUBROUTINE lammps_set_fix_external_callback(handle, id, funcptr, ptr) &
|
|
|
|
BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_funptr
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id, ptr
|
|
|
|
TYPE(c_funptr), VALUE :: funcptr
|
|
|
|
END SUBROUTINE lammps_set_fix_external_callback
|
2022-11-30 08:28:52 +08:00
|
|
|
|
|
|
|
FUNCTION lammps_fix_external_get_force(handle, id) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id
|
|
|
|
TYPE(c_ptr) :: lammps_fix_external_get_force
|
|
|
|
END FUNCTION lammps_fix_external_get_force
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-11-30 13:23:14 +08:00
|
|
|
SUBROUTINE lammps_fix_external_set_energy_global(handle, id, eng) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_double
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id
|
|
|
|
REAL(c_double), VALUE :: eng
|
|
|
|
END SUBROUTINE lammps_fix_external_set_energy_global
|
|
|
|
|
|
|
|
SUBROUTINE lammps_fix_external_set_virial_global(handle, id, virial) &
|
|
|
|
BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id, virial
|
|
|
|
END SUBROUTINE lammps_fix_external_set_virial_global
|
|
|
|
|
|
|
|
SUBROUTINE lammps_fix_external_set_energy_peratom(handle, id, eng) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id, eng
|
|
|
|
END SUBROUTINE lammps_fix_external_set_energy_peratom
|
|
|
|
|
|
|
|
SUBROUTINE lammps_fix_external_set_virial_peratom(handle, id, virial) &
|
|
|
|
BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id, virial
|
|
|
|
END SUBROUTINE lammps_fix_external_set_virial_peratom
|
|
|
|
|
|
|
|
SUBROUTINE lammps_fix_external_set_vector_length(handle, id, length) &
|
|
|
|
BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id
|
|
|
|
INTEGER(c_int), VALUE :: length
|
|
|
|
END SUBROUTINE lammps_fix_external_set_vector_length
|
|
|
|
|
|
|
|
SUBROUTINE lammps_fix_external_set_vector(handle, id, idx, val) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int, c_double
|
|
|
|
TYPE(c_ptr), VALUE :: handle, id
|
|
|
|
INTEGER(c_int), VALUE :: idx
|
|
|
|
REAL(c_double), VALUE :: val
|
|
|
|
END SUBROUTINE lammps_fix_external_set_vector
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE lammps_flush_buffers(handle) BIND(C)
|
2022-10-04 11:49:17 +08:00
|
|
|
IMPORT :: c_ptr
|
2022-09-28 20:40:06 +08:00
|
|
|
IMPLICIT NONE
|
2022-10-04 11:49:17 +08:00
|
|
|
TYPE(c_ptr), VALUE :: handle
|
2022-09-28 20:40:06 +08:00
|
|
|
END SUBROUTINE lammps_flush_buffers
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-09-15 10:07:32 +08:00
|
|
|
SUBROUTINE lammps_free(ptr) BIND(C)
|
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: ptr
|
|
|
|
END SUBROUTINE lammps_free
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-09-15 10:07:32 +08:00
|
|
|
INTEGER(c_int) FUNCTION lammps_is_running(handle) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
END FUNCTION lammps_is_running
|
2022-08-10 10:18:33 +08:00
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE lammps_force_timeout(handle) BIND(C)
|
2022-09-30 08:28:15 +08:00
|
|
|
IMPORT :: c_ptr
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle
|
|
|
|
END SUBROUTINE lammps_force_timeout
|
2020-10-02 22:36:17 +08:00
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
INTEGER(c_int) FUNCTION lammps_has_error(handle) BIND(C)
|
2022-09-27 10:56:03 +08:00
|
|
|
IMPORT :: c_ptr, c_int
|
2022-09-26 21:57:03 +08:00
|
|
|
IMPLICIT NONE
|
2022-09-27 10:56:03 +08:00
|
|
|
TYPE(c_ptr), VALUE :: handle
|
2022-09-26 21:57:03 +08:00
|
|
|
END FUNCTION lammps_has_error
|
2020-08-29 08:56:15 +08:00
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
INTEGER(c_int) FUNCTION lammps_get_last_error_message &
|
2022-09-27 10:56:03 +08:00
|
|
|
(handle, buffer, buf_size) BIND(C)
|
|
|
|
IMPORT :: c_ptr, c_int, c_char
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), VALUE :: handle, buffer
|
|
|
|
INTEGER(c_int), VALUE :: buf_size
|
|
|
|
END FUNCTION lammps_get_last_error_message
|
2022-08-13 00:33:28 +08:00
|
|
|
|
2022-10-21 09:24:59 +08:00
|
|
|
!---------------------------------------------------------------------
|
|
|
|
! Utility functions imported for convenience (not in library.h)
|
|
|
|
!---------------------------------------------------------------------
|
|
|
|
FUNCTION lammps_malloc(size) BIND(C, name='malloc')
|
|
|
|
IMPORT :: c_ptr, c_size_t
|
|
|
|
IMPLICIT NONE
|
|
|
|
INTEGER(c_size_t), VALUE :: size
|
|
|
|
TYPE(c_ptr) :: lammps_malloc
|
|
|
|
END FUNCTION lammps_malloc
|
|
|
|
|
|
|
|
FUNCTION c_strlen(str) BIND(C, name='strlen')
|
|
|
|
IMPORT :: c_ptr, c_size_t
|
|
|
|
IMPLICIT NONE
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: str
|
|
|
|
INTEGER(c_size_t) :: c_strlen
|
|
|
|
END FUNCTION c_strlen
|
|
|
|
|
2020-08-29 08:56:15 +08:00
|
|
|
END INTERFACE
|
|
|
|
|
|
|
|
CONTAINS
|
|
|
|
! Fortran wrappers and helper functions.
|
|
|
|
|
|
|
|
! Constructor for the LAMMPS class.
|
|
|
|
! Combined wrapper around lammps_open_fortran() and lammps_open_no_mpi()
|
2022-08-07 05:43:40 +08:00
|
|
|
TYPE(lammps) FUNCTION lmp_open(args, comm)
|
2022-08-16 10:11:58 +08:00
|
|
|
INTEGER, INTENT(IN), OPTIONAL :: comm
|
2022-08-10 10:18:33 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: args(:)
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE(c_ptr), ALLOCATABLE :: argv(:)
|
2022-08-07 05:43:40 +08:00
|
|
|
INTEGER(c_int) :: i, c_comm, argc
|
2020-08-29 08:56:15 +08:00
|
|
|
|
|
|
|
IF (PRESENT(args)) THEN
|
2022-11-30 05:37:15 +08:00
|
|
|
! convert fortran argument list to c style
|
|
|
|
argc = SIZE(args)
|
|
|
|
ALLOCATE(argv(argc))
|
|
|
|
DO i=1, argc
|
|
|
|
argv(i) = f2c_string(args(i))
|
|
|
|
END DO
|
2020-08-29 08:56:15 +08:00
|
|
|
ELSE
|
2022-11-30 05:37:15 +08:00
|
|
|
argc = 1
|
|
|
|
ALLOCATE(argv(1))
|
|
|
|
argv(1) = f2c_string("liblammps")
|
2020-08-29 08:56:15 +08:00
|
|
|
ENDIF
|
|
|
|
|
|
|
|
IF (PRESENT(comm)) THEN
|
2022-11-30 05:37:15 +08:00
|
|
|
c_comm = comm
|
|
|
|
lmp_open%handle = lammps_open(argc, argv, c_comm)
|
2020-08-29 08:56:15 +08:00
|
|
|
ELSE
|
2022-11-30 05:37:15 +08:00
|
|
|
lmp_open%handle = lammps_open_no_mpi(argc, argv, c_null_ptr)
|
2020-08-29 08:56:15 +08:00
|
|
|
END IF
|
|
|
|
|
|
|
|
! Clean up allocated memory
|
2022-08-07 05:43:40 +08:00
|
|
|
DO i=1, argc
|
2022-11-30 05:37:15 +08:00
|
|
|
CALL lammps_free(argv(i))
|
2020-08-29 08:56:15 +08:00
|
|
|
END DO
|
|
|
|
DEALLOCATE(argv)
|
2022-09-24 05:23:51 +08:00
|
|
|
|
|
|
|
! Assign style and type members so lmp_open%style%global and such work
|
|
|
|
lmp_open%style%global = LMP_STYLE_GLOBAL
|
|
|
|
lmp_open%style%atom = LMP_STYLE_ATOM
|
|
|
|
lmp_open%style%local = LMP_STYLE_LOCAL
|
|
|
|
lmp_open%type%scalar = LMP_TYPE_SCALAR
|
|
|
|
lmp_open%type%vector = LMP_TYPE_VECTOR
|
|
|
|
lmp_open%type%array = LMP_TYPE_ARRAY
|
2023-06-09 07:12:59 +08:00
|
|
|
lmp_open%dtype%i32 = LAMMPS_INT
|
|
|
|
lmp_open%dtype%i64 = LAMMPS_INT64
|
|
|
|
lmp_open%dtype%r64 = LAMMPS_DOUBLE
|
|
|
|
lmp_open%dtype%str = LAMMPS_STRING
|
2022-11-25 11:07:46 +08:00
|
|
|
|
|
|
|
! Assign constants for bigint and tagint for use elsewhere
|
|
|
|
SIZE_TAGINT = lmp_extract_setting(lmp_open, 'tagint')
|
|
|
|
SIZE_BIGINT = lmp_extract_setting(lmp_open, 'bigint')
|
|
|
|
SIZE_IMAGEINT = lmp_extract_setting(lmp_open, 'imageint')
|
2020-08-29 08:56:15 +08:00
|
|
|
END FUNCTION lmp_open
|
|
|
|
|
|
|
|
! Combined Fortran wrapper around lammps_close() and lammps_mpi_finalize()
|
2022-08-07 05:43:40 +08:00
|
|
|
SUBROUTINE lmp_close(self, finalize)
|
2022-09-24 05:23:51 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-08-10 10:18:33 +08:00
|
|
|
LOGICAL, INTENT(IN), OPTIONAL :: finalize
|
2020-08-29 08:56:15 +08:00
|
|
|
|
|
|
|
CALL lammps_close(self%handle)
|
|
|
|
|
|
|
|
IF (PRESENT(finalize)) THEN
|
2022-11-30 05:37:15 +08:00
|
|
|
IF (finalize) THEN
|
|
|
|
CALL lammps_kokkos_finalize()
|
|
|
|
CALL lammps_mpi_finalize()
|
|
|
|
END IF
|
2020-08-29 08:56:15 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_close
|
|
|
|
|
2022-09-24 04:28:15 +08:00
|
|
|
! equivalent function to lammps_error()
|
|
|
|
SUBROUTINE lmp_error(self, error_type, error_text)
|
|
|
|
CLASS(lammps) :: self
|
2022-10-22 02:44:39 +08:00
|
|
|
INTEGER(c_int) :: error_type
|
2022-09-24 04:28:15 +08:00
|
|
|
CHARACTER(len=*) :: error_text
|
|
|
|
TYPE(c_ptr) :: str
|
|
|
|
|
|
|
|
str = f2c_string(error_text)
|
|
|
|
CALL lammps_error(self%handle, error_type, str)
|
|
|
|
CALL lammps_free(str)
|
|
|
|
END SUBROUTINE lmp_error
|
|
|
|
|
2024-09-20 15:46:36 +08:00
|
|
|
! equivalent function to lammps_expand()
|
|
|
|
FUNCTION lmp_expand(self, line)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(len=*), INTENT(IN) :: line
|
|
|
|
TYPE(c_ptr) :: str, res
|
|
|
|
CHARACTER(len=:), ALLOCATABLE :: lmp_expand
|
|
|
|
|
|
|
|
str = f2c_string(line)
|
|
|
|
res = lammps_expand(self%handle, str)
|
|
|
|
CALL lammps_free(str)
|
|
|
|
lmp_expand = c2f_string(res)
|
|
|
|
CALL lammps_free(res)
|
|
|
|
END FUNCTION lmp_expand
|
|
|
|
|
2022-08-13 00:33:28 +08:00
|
|
|
! equivalent function to lammps_file()
|
2022-08-07 05:43:40 +08:00
|
|
|
SUBROUTINE lmp_file(self, filename)
|
2022-09-24 05:23:51 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2024-09-20 15:46:36 +08:00
|
|
|
CHARACTER(len=*), INTENT(IN) :: filename
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE(c_ptr) :: str
|
|
|
|
|
|
|
|
str = f2c_string(filename)
|
2022-08-07 05:43:40 +08:00
|
|
|
CALL lammps_file(self%handle, str)
|
2020-08-29 08:56:15 +08:00
|
|
|
CALL lammps_free(str)
|
|
|
|
END SUBROUTINE lmp_file
|
|
|
|
|
|
|
|
! equivalent function to lammps_command()
|
2022-08-07 05:43:40 +08:00
|
|
|
SUBROUTINE lmp_command(self, cmd)
|
2022-09-24 05:23:51 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2024-09-20 15:46:36 +08:00
|
|
|
CHARACTER(len=*), INTENT(IN) :: cmd
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE(c_ptr) :: str
|
|
|
|
|
|
|
|
str = f2c_string(cmd)
|
2022-08-07 05:43:40 +08:00
|
|
|
CALL lammps_command(self%handle, str)
|
2020-08-29 08:56:15 +08:00
|
|
|
CALL lammps_free(str)
|
|
|
|
END SUBROUTINE lmp_command
|
|
|
|
|
|
|
|
! equivalent function to lammps_commands_list()
|
2022-08-07 05:43:40 +08:00
|
|
|
SUBROUTINE lmp_commands_list(self, cmds)
|
2022-09-24 05:23:51 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-08-16 10:11:58 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: cmds(:)
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE(c_ptr), ALLOCATABLE :: cmdv(:)
|
2022-08-07 05:43:40 +08:00
|
|
|
INTEGER :: i, ncmd
|
2020-08-29 08:56:15 +08:00
|
|
|
|
|
|
|
! convert command list to c style
|
|
|
|
ncmd = SIZE(cmds)
|
|
|
|
ALLOCATE(cmdv(ncmd))
|
2022-08-07 05:43:40 +08:00
|
|
|
DO i=1, ncmd
|
2020-08-29 08:56:15 +08:00
|
|
|
cmdv(i) = f2c_string(cmds(i))
|
|
|
|
END DO
|
|
|
|
|
2022-08-07 05:43:40 +08:00
|
|
|
CALL lammps_commands_list(self%handle, ncmd, cmdv)
|
2020-08-29 08:56:15 +08:00
|
|
|
|
|
|
|
! Clean up allocated memory
|
2022-08-07 05:43:40 +08:00
|
|
|
DO i=1, ncmd
|
2020-08-29 08:56:15 +08:00
|
|
|
CALL lammps_free(cmdv(i))
|
|
|
|
END DO
|
|
|
|
DEALLOCATE(cmdv)
|
|
|
|
END SUBROUTINE lmp_commands_list
|
|
|
|
|
|
|
|
! equivalent function to lammps_commands_string()
|
2022-08-07 05:43:40 +08:00
|
|
|
SUBROUTINE lmp_commands_string(self, str)
|
2022-09-24 05:23:51 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2024-09-20 15:46:36 +08:00
|
|
|
CHARACTER(len=*), INTENT(IN) :: str
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE(c_ptr) :: tmp
|
|
|
|
|
|
|
|
tmp = f2c_string(str)
|
2022-08-07 05:43:40 +08:00
|
|
|
CALL lammps_commands_string(self%handle, tmp)
|
2020-08-29 08:56:15 +08:00
|
|
|
CALL lammps_free(tmp)
|
|
|
|
END SUBROUTINE lmp_commands_string
|
|
|
|
|
2022-08-10 10:18:33 +08:00
|
|
|
! equivalent function to lammps_get_natoms
|
2022-10-01 07:50:16 +08:00
|
|
|
REAL(c_double) FUNCTION lmp_get_natoms(self)
|
2022-08-10 10:18:33 +08:00
|
|
|
CLASS(lammps) :: self
|
|
|
|
|
|
|
|
lmp_get_natoms = lammps_get_natoms(self%handle)
|
|
|
|
END FUNCTION lmp_get_natoms
|
|
|
|
|
|
|
|
! equivalent function to lammps_get_thermo
|
2022-10-01 07:50:16 +08:00
|
|
|
REAL(c_double) FUNCTION lmp_get_thermo(self,name)
|
2022-08-10 10:18:33 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2024-09-20 15:46:36 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
2022-10-04 11:49:17 +08:00
|
|
|
TYPE(c_ptr) :: Cname
|
2022-08-10 10:18:33 +08:00
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
lmp_get_thermo = lammps_get_thermo(self%handle, Cname)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END FUNCTION lmp_get_thermo
|
|
|
|
|
2023-06-09 07:12:59 +08:00
|
|
|
! equivalent function to lammps_last_thermo
|
|
|
|
FUNCTION lmp_last_thermo(self,what,index) RESULT(thermo_data)
|
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: what
|
2024-09-20 15:46:36 +08:00
|
|
|
INTEGER, INTENT(IN) :: index
|
2023-06-11 09:36:56 +08:00
|
|
|
INTEGER(c_int) :: idx
|
2023-06-09 07:12:59 +08:00
|
|
|
TYPE(lammps_data) :: thermo_data, type_data
|
|
|
|
INTEGER(c_int) :: datatype
|
|
|
|
TYPE(c_ptr) :: Cname, Cptr
|
|
|
|
|
2023-06-11 09:36:56 +08:00
|
|
|
idx = index - 1
|
2023-06-09 07:12:59 +08:00
|
|
|
! set data type for known cases
|
|
|
|
SELECT CASE (what)
|
|
|
|
CASE ('step')
|
|
|
|
IF (SIZE_BIGINT == 4_c_int) THEN
|
|
|
|
datatype = LAMMPS_INT
|
|
|
|
ELSE
|
|
|
|
datatype = LAMMPS_INT64
|
|
|
|
END IF
|
|
|
|
CASE ('num')
|
|
|
|
datatype = LAMMPS_INT
|
|
|
|
CASE ('type')
|
|
|
|
datatype = LAMMPS_INT
|
|
|
|
CASE ('keyword')
|
|
|
|
datatype = LAMMPS_STRING
|
|
|
|
CASE ('data')
|
|
|
|
Cname = f2c_string('type')
|
2023-06-11 09:36:56 +08:00
|
|
|
Cptr = lammps_last_thermo(self%handle,Cname,idx)
|
2023-06-09 07:12:59 +08:00
|
|
|
type_data%lammps_instance => self
|
|
|
|
type_data%datatype = DATA_INT
|
|
|
|
CALL C_F_POINTER(Cptr, type_data%i32)
|
|
|
|
datatype = type_data%i32
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
CASE DEFAULT
|
|
|
|
datatype = -1
|
|
|
|
END SELECT
|
|
|
|
|
|
|
|
Cname = f2c_string(what)
|
2023-06-11 09:36:56 +08:00
|
|
|
Cptr = lammps_last_thermo(self%handle,Cname,idx)
|
2023-06-09 07:12:59 +08:00
|
|
|
CALL lammps_free(Cname)
|
|
|
|
|
|
|
|
thermo_data%lammps_instance => self
|
|
|
|
SELECT CASE (datatype)
|
|
|
|
CASE (LAMMPS_INT)
|
|
|
|
thermo_data%datatype = DATA_INT
|
|
|
|
CALL C_F_POINTER(Cptr, thermo_data%i32)
|
|
|
|
CASE (LAMMPS_INT64)
|
|
|
|
thermo_data%datatype = DATA_INT64
|
|
|
|
CALL C_F_POINTER(Cptr, thermo_data%i64)
|
|
|
|
CASE (LAMMPS_DOUBLE)
|
|
|
|
thermo_data%datatype = DATA_DOUBLE
|
|
|
|
CALL C_F_POINTER(Cptr, thermo_data%r64)
|
|
|
|
CASE (LAMMPS_STRING)
|
|
|
|
thermo_data%datatype = DATA_STRING
|
|
|
|
thermo_data%str = c2f_string(Cptr)
|
|
|
|
CASE DEFAULT
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Unknown pointer type in last_thermo')
|
|
|
|
END SELECT
|
|
|
|
END FUNCTION lmp_last_thermo
|
|
|
|
|
2022-08-10 10:18:33 +08:00
|
|
|
! equivalent subroutine to lammps_extract_box
|
2022-09-24 04:28:15 +08:00
|
|
|
SUBROUTINE lmp_extract_box(self, boxlo, boxhi, xy, yz, xz, pflags, boxflag)
|
2022-08-10 10:18:33 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
REAL(c_double), INTENT(OUT), TARGET, OPTIONAL :: boxlo(3), boxhi(3)
|
|
|
|
REAL(c_double), INTENT(OUT), TARGET, OPTIONAL :: xy, yz, xz
|
|
|
|
LOGICAL, INTENT(OUT), OPTIONAL :: pflags(3), boxflag
|
2022-10-04 11:49:17 +08:00
|
|
|
INTEGER(c_int), TARGET :: c_pflags(3), c_boxflag
|
2022-10-01 07:50:16 +08:00
|
|
|
TYPE(c_ptr) :: ptr(7)
|
2022-08-10 10:18:33 +08:00
|
|
|
|
|
|
|
ptr = c_null_ptr
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(boxlo)) ptr(1) = C_LOC(boxlo(1))
|
|
|
|
IF (PRESENT(boxhi)) ptr(2) = C_LOC(boxhi(1))
|
|
|
|
IF (PRESENT(xy)) ptr(3) = C_LOC(xy)
|
|
|
|
IF (PRESENT(yz)) ptr(4) = C_LOC(yz)
|
|
|
|
IF (PRESENT(xz)) ptr(5) = C_LOC(xz)
|
|
|
|
IF (PRESENT(pflags)) ptr(6) = C_LOC(c_pflags(1))
|
|
|
|
IF (PRESENT(boxflag)) ptr(7) = C_LOC(c_boxflag)
|
2022-08-10 10:18:33 +08:00
|
|
|
CALL lammps_extract_box(self%handle, ptr(1), ptr(2), ptr(3), ptr(4), &
|
|
|
|
ptr(5), ptr(6), ptr(7))
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(pflags)) pflags = (c_pflags /= 0_c_int)
|
|
|
|
IF (PRESENT(boxflag)) boxflag = (c_boxflag /= 0_c_int)
|
2022-08-10 10:18:33 +08:00
|
|
|
END SUBROUTINE lmp_extract_box
|
|
|
|
|
2022-08-10 13:55:17 +08:00
|
|
|
! equivalent function to lammps_reset_box
|
2022-09-24 04:28:15 +08:00
|
|
|
SUBROUTINE lmp_reset_box(self, boxlo, boxhi, xy, yz, xz)
|
2022-08-10 13:55:17 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-10-04 11:49:17 +08:00
|
|
|
REAL(c_double), INTENT(IN) :: boxlo(3), boxhi(3), xy, yz, xz
|
2022-08-10 13:55:17 +08:00
|
|
|
|
2022-09-24 04:28:15 +08:00
|
|
|
CALL lammps_reset_box(self%handle, boxlo, boxhi, xy, yz, xz)
|
2022-08-10 13:55:17 +08:00
|
|
|
END SUBROUTINE lmp_reset_box
|
|
|
|
|
|
|
|
! equivalent function to lammps_memory_usage
|
|
|
|
SUBROUTINE lmp_memory_usage(self,meminfo)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER, PARAMETER :: MEMINFO_ELEM = 3
|
2022-10-01 07:50:16 +08:00
|
|
|
REAL(c_double), DIMENSION(MEMINFO_ELEM), INTENT(OUT) :: meminfo
|
2022-08-10 13:55:17 +08:00
|
|
|
|
|
|
|
CALL lammps_memory_usage(self%handle,meminfo)
|
|
|
|
END SUBROUTINE lmp_memory_usage
|
|
|
|
|
|
|
|
! equivalent function to lammps_get_mpi_comm
|
2022-09-24 04:28:15 +08:00
|
|
|
INTEGER FUNCTION lmp_get_mpi_comm(self)
|
2022-08-10 13:55:17 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
|
|
|
|
lmp_get_mpi_comm = lammps_get_mpi_comm(self%handle)
|
|
|
|
END FUNCTION lmp_get_mpi_comm
|
|
|
|
|
|
|
|
! equivalent function to lammps_extract_setting
|
2022-10-01 07:50:16 +08:00
|
|
|
INTEGER(c_int) FUNCTION lmp_extract_setting(self, keyword)
|
2022-08-10 13:55:17 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: keyword
|
|
|
|
TYPE(c_ptr) :: Ckeyword
|
|
|
|
|
|
|
|
Ckeyword = f2c_string(keyword)
|
|
|
|
lmp_extract_setting = lammps_extract_setting(self%handle, Ckeyword)
|
|
|
|
CALL lammps_free(Ckeyword)
|
|
|
|
END FUNCTION lmp_extract_setting
|
|
|
|
|
2022-08-12 13:04:25 +08:00
|
|
|
! equivalent function to lammps_extract_global
|
|
|
|
! the assignment is actually overloaded so as to bind the pointers to
|
|
|
|
! lammps data based on the information available from LAMMPS
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lmp_extract_global(self, name) RESULT(global_data)
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
2022-08-12 13:04:25 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
TYPE(lammps_data) :: global_data
|
|
|
|
INTEGER(c_int) :: datatype
|
|
|
|
TYPE(c_ptr) :: Cname, Cptr
|
2022-10-31 05:42:10 +08:00
|
|
|
INTEGER(c_size_t) :: length
|
2022-08-12 13:04:25 +08:00
|
|
|
|
|
|
|
! Determine vector length
|
|
|
|
! FIXME Is there a way to get the length of the vector from C rather
|
|
|
|
! than defining it here AND in the Python API?
|
|
|
|
SELECT CASE (name)
|
|
|
|
CASE ('boxlo','boxhi','sublo','subhi','sublo_lambda','subhi_lambda', &
|
|
|
|
'periodicity')
|
|
|
|
length = 3
|
|
|
|
CASE DEFAULT
|
|
|
|
length = 1
|
2022-11-30 05:37:15 +08:00
|
|
|
! string cases do not use "length"
|
2022-08-12 13:04:25 +08:00
|
|
|
END SELECT
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
datatype = lammps_extract_global_datatype(self%handle, Cname)
|
|
|
|
! above could be c_null_ptr in place of self%handle...doesn't matter
|
|
|
|
Cptr = lammps_extract_global(self%handle, Cname)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
global_data%lammps_instance => self
|
2022-08-12 13:04:25 +08:00
|
|
|
SELECT CASE (datatype)
|
|
|
|
CASE (LAMMPS_INT)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (length == 1) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
global_data%datatype = DATA_INT
|
|
|
|
CALL C_F_POINTER(Cptr, global_data%i32)
|
|
|
|
ELSE
|
|
|
|
global_data%datatype = DATA_INT_1D
|
|
|
|
CALL C_F_POINTER(Cptr, global_data%i32_vec, [length])
|
|
|
|
END IF
|
|
|
|
CASE (LAMMPS_INT64)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (length == 1) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
global_data%datatype = DATA_INT64
|
|
|
|
CALL C_F_POINTER(Cptr, global_data%i64)
|
|
|
|
ELSE
|
|
|
|
global_data%datatype = DATA_INT64_1D
|
|
|
|
CALL C_F_POINTER(Cptr, global_data%i64_vec, [length])
|
|
|
|
END IF
|
|
|
|
CASE (LAMMPS_DOUBLE)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (length == 1) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
global_data%datatype = DATA_DOUBLE
|
|
|
|
CALL C_F_POINTER(Cptr, global_data%r64)
|
|
|
|
ELSE
|
|
|
|
global_data%datatype = DATA_DOUBLE_1D
|
|
|
|
CALL C_F_POINTER(Cptr, global_data%r64_vec, [length])
|
|
|
|
END IF
|
|
|
|
CASE (LAMMPS_STRING)
|
|
|
|
global_data%datatype = DATA_STRING
|
2022-10-26 08:51:08 +08:00
|
|
|
global_data%str = c2f_string(Cptr)
|
2022-09-26 12:54:18 +08:00
|
|
|
CASE DEFAULT
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Unknown pointer type in extract_global')
|
2022-08-12 13:04:25 +08:00
|
|
|
END SELECT
|
|
|
|
END FUNCTION
|
|
|
|
|
2024-07-18 19:39:15 +08:00
|
|
|
! equivalent function to lammps_extract_pair_dimension
|
|
|
|
FUNCTION lmp_extract_pair_dimension(self, name) RESULT(dim)
|
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int) :: dim
|
|
|
|
TYPE(c_ptr) :: Cname
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
dim = lammps_extract_pair_dimension(self%handle, Cname)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END FUNCTION
|
|
|
|
|
|
|
|
! equivalent function to lammps_extract_pair
|
|
|
|
! the assignment is actually overloaded so as to bind the pointers to
|
|
|
|
! lammps data based on the information available from LAMMPS
|
|
|
|
FUNCTION lmp_extract_pair(self, name) RESULT(pair_data)
|
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
TYPE(lammps_data) :: pair_data
|
|
|
|
INTEGER(c_int) :: dim
|
|
|
|
TYPE(c_ptr) :: Cname, Cptr
|
2024-07-19 01:37:04 +08:00
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: Carrayptr
|
2024-07-18 19:39:15 +08:00
|
|
|
INTEGER(c_size_t) :: length
|
|
|
|
|
|
|
|
! Determine extracted arrays length and dimension
|
|
|
|
length = lmp_extract_setting(self, 'ntypes') + 1
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
dim = lammps_extract_pair_dimension(self%handle, Cname)
|
|
|
|
! above could be c_null_ptr in place of self%handle...doesn't matter
|
|
|
|
Cptr = lammps_extract_pair(self%handle, Cname)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
|
|
|
|
pair_data%lammps_instance => self
|
|
|
|
SELECT CASE (dim)
|
|
|
|
CASE (0)
|
|
|
|
pair_data%datatype = DATA_DOUBLE
|
|
|
|
CALL C_F_POINTER(Cptr, pair_data%r64)
|
|
|
|
CASE (1)
|
|
|
|
pair_data%datatype = DATA_DOUBLE_1D
|
|
|
|
CALL C_F_POINTER(Cptr, pair_data%r64_vec, [length])
|
|
|
|
CASE (2)
|
|
|
|
pair_data%datatype = DATA_DOUBLE_2D
|
2024-07-19 01:37:04 +08:00
|
|
|
! First, we dereference the void** pointer to point to the void*
|
|
|
|
CALL C_F_POINTER(Cptr, Carrayptr, [length])
|
|
|
|
! Catomptr(1) now points to the first element of the array
|
|
|
|
CALL C_F_POINTER(Carrayptr(1), pair_data%r64_mat, [length, length])
|
|
|
|
CASE DEFAULT
|
2024-07-18 19:39:15 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2024-07-19 01:37:04 +08:00
|
|
|
'Unsupported dimension or unknown property name in extract_pair')
|
2024-07-18 19:39:15 +08:00
|
|
|
END SELECT
|
|
|
|
END FUNCTION
|
|
|
|
|
2024-05-25 19:01:28 +08:00
|
|
|
! equivalent function to lammps_map_atom (for 32-bit integer tags)
|
|
|
|
INTEGER FUNCTION lmp_map_atom_int(self, id)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), INTENT(IN), TARGET :: id
|
|
|
|
INTEGER(c_int64_t), TARGET :: id64
|
|
|
|
TYPE(c_ptr) :: Cptr
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT == 8) THEN
|
|
|
|
id64 = id
|
|
|
|
Cptr = C_LOC(id64)
|
|
|
|
ELSE
|
|
|
|
Cptr = C_LOC(id)
|
|
|
|
END IF
|
|
|
|
lmp_map_atom_int = lammps_map_atom(self%handle, Cptr) + 1
|
|
|
|
END FUNCTION lmp_map_atom_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_map_atom (for 64-bit integer tags)
|
|
|
|
INTEGER FUNCTION lmp_map_atom_big(self, id)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int64_t), INTENT(IN), TARGET :: id
|
|
|
|
INTEGER(c_int), TARGET :: id32
|
|
|
|
TYPE(c_ptr) :: Cptr
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT == 8) THEN
|
|
|
|
Cptr = C_LOC(id)
|
|
|
|
ELSE
|
2024-09-07 02:00:27 +08:00
|
|
|
id32 = INT(id, c_int)
|
2024-05-25 19:01:28 +08:00
|
|
|
Cptr = C_LOC(id32)
|
|
|
|
END IF
|
|
|
|
lmp_map_atom_big = lammps_map_atom(self%handle, Cptr) + 1
|
|
|
|
END FUNCTION lmp_map_atom_big
|
2024-05-25 19:11:02 +08:00
|
|
|
|
2022-09-23 08:16:15 +08:00
|
|
|
! equivalent function to lammps_extract_atom
|
|
|
|
! the assignment is actually overloaded so as to bind the pointers to
|
|
|
|
! lammps data based on the information available from LAMMPS
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lmp_extract_atom(self, name) RESULT(peratom_data)
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
2022-09-23 08:16:15 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
TYPE(lammps_data) :: peratom_data
|
|
|
|
|
|
|
|
INTEGER(c_int) :: datatype
|
|
|
|
TYPE(c_ptr) :: Cname, Cptr
|
|
|
|
INTEGER(c_int) :: ntypes, nmax
|
|
|
|
INTEGER :: nrows, ncols
|
|
|
|
REAL(c_double), DIMENSION(:), POINTER :: dummy
|
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: Catomptr
|
2022-09-26 12:54:18 +08:00
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: error_msg
|
2022-09-23 08:16:15 +08:00
|
|
|
|
|
|
|
nmax = lmp_extract_setting(self, 'nmax')
|
|
|
|
ntypes = lmp_extract_setting(self, 'ntypes')
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
datatype = lammps_extract_atom_datatype(self%handle, Cname)
|
2024-08-31 11:16:37 +08:00
|
|
|
nrows = lammps_extract_atom_size(self%handle, Cname, LMP_SIZE_ROWS)
|
|
|
|
ncols = lammps_extract_atom_size(self%handle, Cname, LMP_SIZE_COLS)
|
2022-09-23 08:16:15 +08:00
|
|
|
Cptr = lammps_extract_atom(self%handle, Cname)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
peratom_data%lammps_instance => self
|
2022-09-23 08:16:15 +08:00
|
|
|
SELECT CASE (datatype)
|
|
|
|
CASE (LAMMPS_INT)
|
|
|
|
peratom_data%datatype = DATA_INT_1D
|
2024-08-31 11:16:37 +08:00
|
|
|
CALL C_F_POINTER(Cptr, peratom_data%i32_vec, [nrows])
|
2022-09-23 08:16:15 +08:00
|
|
|
CASE (LAMMPS_INT64)
|
|
|
|
peratom_data%datatype = DATA_INT64_1D
|
2024-08-31 11:16:37 +08:00
|
|
|
CALL C_F_POINTER(Cptr, peratom_data%i64_vec, [nrows])
|
2022-09-23 08:16:15 +08:00
|
|
|
CASE (LAMMPS_DOUBLE)
|
|
|
|
peratom_data%datatype = DATA_DOUBLE_1D
|
2024-08-31 10:50:42 +08:00
|
|
|
! The mass array is allocated from 0, but only used from 1. We also want to use it from 1.
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (name == 'mass') THEN
|
2024-08-31 11:16:37 +08:00
|
|
|
CALL C_F_POINTER(Cptr, dummy, [nrows])
|
2022-09-23 08:16:15 +08:00
|
|
|
peratom_data%r64_vec(0:) => dummy
|
|
|
|
ELSE
|
2024-08-31 11:16:37 +08:00
|
|
|
CALL C_F_POINTER(Cptr, peratom_data%r64_vec, [nrows])
|
2022-09-23 08:16:15 +08:00
|
|
|
END IF
|
|
|
|
CASE (LAMMPS_DOUBLE_2D)
|
|
|
|
peratom_data%datatype = DATA_DOUBLE_2D
|
|
|
|
! First, we dereference the void** pointer to point to the void*
|
2024-08-31 11:16:37 +08:00
|
|
|
CALL C_F_POINTER(Cptr, Catomptr, [nrows])
|
2022-09-23 08:16:15 +08:00
|
|
|
! Catomptr(1) now points to the first element of the array
|
2024-09-01 00:16:27 +08:00
|
|
|
! rows and columns are swapped in Fortran
|
2024-08-31 11:16:37 +08:00
|
|
|
CALL C_F_POINTER(Catomptr(1), peratom_data%r64_mat, [ncols,nrows])
|
2022-09-23 08:16:15 +08:00
|
|
|
CASE (-1)
|
2022-09-26 12:54:18 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2022-12-23 12:23:32 +08:00
|
|
|
'per-atom property ' // name // ' not found in extract_setting')
|
2022-09-23 08:16:15 +08:00
|
|
|
CASE DEFAULT
|
2022-12-23 12:23:32 +08:00
|
|
|
error_msg = ' '
|
2022-09-26 12:54:18 +08:00
|
|
|
WRITE(error_msg,'(A,I0,A)') 'return value ', datatype, &
|
|
|
|
' from lammps_extract_atom_datatype not known [Fortran/extract_atom]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
2022-09-23 08:16:15 +08:00
|
|
|
END SELECT
|
|
|
|
END FUNCTION lmp_extract_atom
|
|
|
|
|
2022-09-24 05:23:51 +08:00
|
|
|
! equivalent function to lammps_extract_compute
|
|
|
|
! the assignment operator is overloaded so as to bind the pointers to
|
|
|
|
! lammps data based on the information available from LAMMPS
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lmp_extract_compute(self, id, style, type) RESULT(compute_data)
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
2022-09-24 05:23:51 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
INTEGER(c_int), INTENT(IN) :: style, type
|
|
|
|
TYPE(lammps_data) :: compute_data
|
|
|
|
|
|
|
|
TYPE(c_ptr) :: Cid, Cptr, Ctemp
|
|
|
|
INTEGER :: nrows, ncols, length
|
|
|
|
INTEGER(c_int), POINTER :: temp
|
2022-09-26 12:54:18 +08:00
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: Ccomputeptr
|
2022-09-24 05:23:51 +08:00
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
Cptr = lammps_extract_compute(self%handle, Cid, style, type)
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (.NOT. C_ASSOCIATED(Cptr)) THEN
|
2022-09-26 12:54:18 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Pointer from LAMMPS is NULL [Fortran/extract_compute]')
|
|
|
|
END IF
|
|
|
|
|
2022-09-24 05:23:51 +08:00
|
|
|
! Remember that rows and columns in C are transposed in Fortran!
|
2022-09-29 11:09:26 +08:00
|
|
|
compute_data%lammps_instance => self
|
2022-09-24 05:23:51 +08:00
|
|
|
SELECT CASE (type)
|
|
|
|
CASE (LMP_TYPE_SCALAR)
|
2022-09-26 12:54:18 +08:00
|
|
|
compute_data%datatype = DATA_DOUBLE
|
2022-09-24 05:23:51 +08:00
|
|
|
length = 1
|
|
|
|
nrows = 1
|
|
|
|
ncols = 1
|
|
|
|
CALL C_F_POINTER(Cptr, compute_data%r64)
|
|
|
|
CASE (LMP_TYPE_VECTOR)
|
2022-09-26 12:54:18 +08:00
|
|
|
compute_data%datatype = DATA_DOUBLE_1D
|
2022-09-24 05:23:51 +08:00
|
|
|
IF (style == LMP_STYLE_ATOM) THEN
|
|
|
|
length = self%extract_setting('nmax')
|
|
|
|
ELSE
|
|
|
|
Ctemp = lammps_extract_compute(self%handle,Cid,style,LMP_SIZE_VECTOR)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
length = temp
|
|
|
|
END IF
|
|
|
|
CALL C_F_POINTER(Cptr, compute_data%r64_vec, [length])
|
|
|
|
CASE (LMP_TYPE_ARRAY)
|
2022-09-26 12:54:18 +08:00
|
|
|
compute_data%datatype = DATA_DOUBLE_2D
|
2022-09-24 05:23:51 +08:00
|
|
|
IF (style == LMP_STYLE_ATOM) THEN
|
2022-09-26 12:54:18 +08:00
|
|
|
ncols = self%extract_setting('nmax')
|
|
|
|
Ctemp = lammps_extract_compute(self%handle,Cid,style,LMP_SIZE_COLS)
|
2022-09-24 05:23:51 +08:00
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
2022-09-26 12:54:18 +08:00
|
|
|
nrows = temp
|
2022-09-24 05:23:51 +08:00
|
|
|
ELSE
|
|
|
|
Ctemp = lammps_extract_compute(self%handle,Cid,style,LMP_SIZE_ROWS)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
ncols = temp
|
|
|
|
Ctemp = lammps_extract_compute(self%handle,Cid,style,LMP_SIZE_COLS)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
nrows = temp
|
|
|
|
END IF
|
2022-09-26 12:54:18 +08:00
|
|
|
! First, we dereference the void** pointer to point to a void* pointer
|
|
|
|
CALL C_F_POINTER(Cptr, Ccomputeptr, [ncols])
|
|
|
|
! Ccomputeptr(1) now points to the first element of the array
|
|
|
|
CALL C_F_POINTER(Ccomputeptr(1), compute_data%r64_mat, [nrows, ncols])
|
2022-09-24 05:23:51 +08:00
|
|
|
CASE DEFAULT
|
2022-09-26 12:54:18 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2022-09-29 11:09:26 +08:00
|
|
|
'unknown type value passed to extract_compute [Fortran API]')
|
2022-09-24 05:23:51 +08:00
|
|
|
END SELECT
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END FUNCTION lmp_extract_compute
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lmp_extract_fix(self, id, style, type, nrow, ncol) RESULT(fix_data)
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
INTEGER(c_int), INTENT(IN) :: style, type
|
|
|
|
INTEGER(c_int), INTENT(IN), OPTIONAL :: nrow, ncol
|
|
|
|
TYPE(lammps_fix_data) :: fix_data
|
|
|
|
|
|
|
|
TYPE(c_ptr) :: Cid, Cptr, Ctemp
|
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: Cfixptr
|
|
|
|
INTEGER(c_int) :: Cnrow, Cncol
|
|
|
|
REAL(c_double), POINTER :: Fptr
|
|
|
|
INTEGER :: nrows, ncols
|
|
|
|
INTEGER(c_int), POINTER :: temp
|
|
|
|
|
|
|
|
! We transpose ncol and nrow so the array appears to be transposed for
|
|
|
|
! global data, as it would be if we could access the C++ array directly
|
|
|
|
Cnrow = -1
|
|
|
|
Cncol = -1
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(nrow)) THEN
|
|
|
|
IF (.NOT. PRESENT(ncol)) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
! Presumably the argument that's there is the vector length
|
|
|
|
Cnrow = nrow - 1_c_int
|
|
|
|
Cncol = -1_c_int
|
|
|
|
ELSE
|
|
|
|
! Otherwise, the array is transposed, so...reverse the indices
|
|
|
|
Cncol = nrow - 1_c_int
|
|
|
|
END IF
|
|
|
|
END IF
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(ncol)) Cnrow = ncol - 1_c_int
|
2022-09-29 11:09:26 +08:00
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
Cptr = lammps_extract_fix(self%handle, Cid, style, type, Cnrow, Cncol)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (.NOT. C_ASSOCIATED(Cptr)) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Pointer from LAMMPS is NULL for fix id "' // id &
|
|
|
|
// '" [Fortran/extract_fix]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
fix_data%lammps_instance => self
|
|
|
|
SELECT CASE (style)
|
|
|
|
CASE (LMP_STYLE_GLOBAL)
|
|
|
|
fix_data%datatype = DATA_DOUBLE
|
|
|
|
CALL C_F_POINTER(Cptr, Fptr)
|
|
|
|
fix_data%r64 = Fptr
|
|
|
|
CALL lammps_free(Cptr)
|
|
|
|
CASE (LMP_STYLE_ATOM, LMP_STYLE_LOCAL)
|
|
|
|
SELECT CASE (type)
|
|
|
|
CASE (LMP_TYPE_SCALAR)
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'There is no such thing as a per-atom or local scalar&
|
|
|
|
& [Fortran/extract_fix]')
|
|
|
|
CASE (LMP_TYPE_VECTOR)
|
|
|
|
fix_data%datatype = DATA_DOUBLE_1D
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (STYLE == LMP_STYLE_ATOM) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
nrows = self%extract_setting('nmax')
|
|
|
|
ELSE
|
|
|
|
Ctemp = lammps_extract_fix(self%handle, Cid, style, &
|
|
|
|
LMP_SIZE_VECTOR, 0_c_int,0_c_int)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
nrows = temp
|
|
|
|
END IF
|
|
|
|
CALL C_F_POINTER(Cptr, fix_data%r64_vec, [nrows])
|
|
|
|
CASE (LMP_TYPE_ARRAY)
|
|
|
|
fix_data%datatype = DATA_DOUBLE_2D
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (STYLE == LMP_STYLE_ATOM) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
! Fortran array is transposed relative to C
|
|
|
|
ncols = self%extract_setting('nmax')
|
|
|
|
Ctemp = lammps_extract_fix(self%handle, Cid, style, &
|
|
|
|
LMP_SIZE_COLS, 0_c_int,0_c_int)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
nrows = temp
|
|
|
|
ELSE
|
|
|
|
! Fortran array is transposed relative to C
|
|
|
|
Ctemp = lammps_extract_fix(self%handle, Cid, style, &
|
|
|
|
LMP_SIZE_COLS, 0_c_int,0_c_int)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
nrows = temp
|
|
|
|
Ctemp = lammps_extract_fix(self%handle, Cid, style, &
|
|
|
|
LMP_SIZE_ROWS, 0_c_int,0_c_int)
|
|
|
|
CALL C_F_POINTER(Ctemp, temp)
|
|
|
|
ncols = temp
|
|
|
|
END IF
|
|
|
|
! First, we dereference the void** to point to a void* pointer
|
|
|
|
CALL C_F_POINTER(Cptr, Cfixptr, [ncols])
|
|
|
|
! Cfixptr(1) now points to the first element of the array
|
|
|
|
CALL C_F_POINTER(Cfixptr(1), fix_data%r64_mat, [nrows, ncols])
|
|
|
|
CASE DEFAULT
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'unknown type value passed to extract_fix [Fortran API]')
|
|
|
|
END SELECT
|
|
|
|
CASE DEFAULT
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'unknown style value passed to extract_fix [Fortran API]')
|
|
|
|
END SELECT
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END FUNCTION lmp_extract_fix
|
|
|
|
|
|
|
|
! equivalent function to lammps_extract_variable
|
2022-10-01 07:50:16 +08:00
|
|
|
FUNCTION lmp_extract_variable(self, name, group) RESULT(variable_data)
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
CHARACTER(LEN=*), INTENT(IN), OPTIONAL :: group
|
|
|
|
TYPE(lammps_variable_data) :: variable_data
|
|
|
|
|
2022-10-01 05:02:26 +08:00
|
|
|
TYPE(c_ptr) :: Cptr, Cname, Cgroup, Cveclength
|
2022-10-31 05:42:10 +08:00
|
|
|
INTEGER(c_size_t) :: length
|
2022-09-29 11:09:26 +08:00
|
|
|
INTEGER(c_int) :: datatype
|
2022-10-01 05:02:26 +08:00
|
|
|
REAL(c_double), POINTER :: double => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:), POINTER :: double_vec => NULL()
|
|
|
|
INTEGER(c_int), POINTER :: Clength => NULL()
|
2022-09-29 11:09:26 +08:00
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(group)) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
Cgroup = f2c_string(group)
|
|
|
|
ELSE
|
|
|
|
Cgroup = c_null_ptr
|
|
|
|
END IF
|
|
|
|
datatype = lammps_extract_variable_datatype(self%handle, Cname)
|
|
|
|
Cptr = lammps_extract_variable(self%handle, Cname, Cgroup)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
CALL lammps_free(Cgroup)
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
variable_data%lammps_instance => self
|
2022-09-29 11:09:26 +08:00
|
|
|
SELECT CASE (datatype)
|
|
|
|
CASE (LMP_VAR_EQUAL)
|
|
|
|
variable_data%datatype = DATA_DOUBLE
|
|
|
|
CALL C_F_POINTER(Cptr, double)
|
|
|
|
variable_data%r64 = double
|
|
|
|
CALL lammps_free(Cptr)
|
|
|
|
CASE (LMP_VAR_ATOM)
|
|
|
|
variable_data%datatype = DATA_DOUBLE_1D
|
|
|
|
length = lmp_extract_setting(self, 'nlocal')
|
|
|
|
CALL C_F_POINTER(Cptr, double_vec, [length])
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(variable_data%r64_vec)) DEALLOCATE(variable_data%r64_vec)
|
|
|
|
ALLOCATE(variable_data%r64_vec(length))
|
2022-09-29 11:09:26 +08:00
|
|
|
variable_data%r64_vec = double_vec
|
|
|
|
CALL lammps_free(Cptr)
|
2022-10-01 05:02:26 +08:00
|
|
|
CASE (LMP_VAR_VECTOR)
|
|
|
|
variable_data%datatype = DATA_DOUBLE_1D
|
|
|
|
Cgroup = f2c_string('LMP_SIZE_VECTOR') ! must match library.cpp
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cveclength = lammps_extract_variable(self%handle, Cname, Cgroup)
|
|
|
|
CALL C_F_POINTER(Cveclength, Clength)
|
|
|
|
length = Clength
|
|
|
|
CALL lammps_free(Cgroup)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
CALL lammps_free(Cveclength)
|
|
|
|
CALL C_F_POINTER(Cptr, double_vec, [length])
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(variable_data%r64_vec)) &
|
2022-10-01 05:02:26 +08:00
|
|
|
DEALLOCATE(variable_data%r64_vec)
|
2022-10-04 11:49:17 +08:00
|
|
|
ALLOCATE(variable_data%r64_vec(length))
|
2022-10-01 05:02:26 +08:00
|
|
|
variable_data%r64_vec = double_vec
|
|
|
|
! DO NOT deallocate the C pointer
|
2022-09-29 11:09:26 +08:00
|
|
|
CASE (LMP_VAR_STRING)
|
|
|
|
variable_data%datatype = DATA_STRING
|
|
|
|
length = c_strlen(Cptr)
|
2022-10-04 11:49:17 +08:00
|
|
|
ALLOCATE(CHARACTER(LEN=length) :: variable_data%str)
|
2022-10-26 08:51:08 +08:00
|
|
|
variable_data%str = c2f_string(Cptr)
|
2022-10-01 05:02:26 +08:00
|
|
|
! DO NOT deallocate the C pointer
|
|
|
|
CASE (-1)
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Variable "' // TRIM(name) // &
|
|
|
|
'" not found [Fortran/extract_variable]')
|
2022-09-29 11:09:26 +08:00
|
|
|
CASE DEFAULT
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Unknown variable type returned from &
|
|
|
|
&lammps_extract_variable_datatype [Fortran/extract_variable]')
|
|
|
|
END SELECT
|
|
|
|
END FUNCTION lmp_extract_variable
|
|
|
|
|
2022-10-01 07:31:18 +08:00
|
|
|
! equivalent function to lammps_set_variable
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE lmp_set_variable(self, name, str)
|
2022-10-01 07:31:18 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-10-01 07:50:16 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name, str
|
2022-10-01 07:31:18 +08:00
|
|
|
INTEGER :: err
|
2022-10-04 11:49:17 +08:00
|
|
|
TYPE(c_ptr) :: Cstr, Cname
|
2022-10-01 07:31:18 +08:00
|
|
|
|
|
|
|
Cstr = f2c_string(str)
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
err = lammps_set_variable(self%handle, Cname, Cstr)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
CALL lammps_free(Cstr)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (err /= 0) THEN
|
2022-10-01 07:31:18 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'WARNING: unable to set string variable "' // name &
|
|
|
|
// '" [Fortran/set_variable]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_set_variable
|
|
|
|
|
2024-01-24 16:51:18 +08:00
|
|
|
! equivalent function to lammps_set_string_variable
|
|
|
|
SUBROUTINE lmp_set_string_variable(self, name, str)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name, str
|
|
|
|
INTEGER :: err
|
|
|
|
TYPE(c_ptr) :: Cstr, Cname
|
|
|
|
|
|
|
|
Cstr = f2c_string(str)
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
err = lammps_set_string_variable(self%handle, Cname, Cstr)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
CALL lammps_free(Cstr)
|
|
|
|
IF (err /= 0) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'WARNING: unable to set string variable "' // name &
|
|
|
|
// '" [Fortran/set_variable]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_set_string_variable
|
|
|
|
|
|
|
|
! equivalent function to lammps_set_internal_variable
|
|
|
|
SUBROUTINE lmp_set_internal_variable(self, name, val)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
REAL(KIND=c_double), INTENT(IN) :: val
|
|
|
|
INTEGER :: err
|
2024-01-27 05:47:22 +08:00
|
|
|
TYPE(c_ptr) :: Cname
|
2024-01-24 16:51:18 +08:00
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
err = lammps_set_internal_variable(self%handle, Cname, val)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
IF (err /= 0) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'WARNING: unable to set internal variable "' // name &
|
|
|
|
// '" [Fortran/set_variable]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_set_internal_variable
|
|
|
|
|
2022-10-03 04:28:10 +08:00
|
|
|
! equivalent function to lammps_gather_atoms (for integers)
|
|
|
|
SUBROUTINE lmp_gather_atoms_int(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
2022-10-03 09:32:42 +08:00
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
2022-10-03 04:28:10 +08:00
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, 'gather_atoms&
|
|
|
|
& requires "count" to be 1 or 3 [Fortran/gather_atoms]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather_atoms with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather_atoms]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-10-03 04:28:10 +08:00
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_atoms(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_atoms_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_atoms (for doubles)
|
|
|
|
SUBROUTINE lmp_gather_atoms_double(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
2022-10-04 11:49:17 +08:00
|
|
|
REAL(c_double) :: dnatoms
|
2022-10-03 09:32:42 +08:00
|
|
|
CHARACTER(LEN=100) :: error_msg
|
2022-10-03 04:28:10 +08:00
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, 'gather_atoms&
|
|
|
|
& requires "count" to be 1 or 3 [Fortran/gather_atoms]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather_atoms with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather_atoms]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-10-03 04:28:10 +08:00
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_atoms(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_atoms_double
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_atoms_concat (for integers)
|
|
|
|
SUBROUTINE lmp_gather_atoms_concat_int(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
2022-10-04 11:49:17 +08:00
|
|
|
REAL(c_double) :: dnatoms
|
2022-10-03 09:32:42 +08:00
|
|
|
CHARACTER(LEN=100) :: error_msg
|
2022-10-03 04:28:10 +08:00
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_atoms_concat requires "count" to be 1 or 3 &
|
|
|
|
&[Fortran/gather_atoms_concat]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather_atoms_concat with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather_atoms_concat]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-10-03 04:28:10 +08:00
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_atoms_concat(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_atoms_concat_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_atoms_concat (for doubles)
|
|
|
|
SUBROUTINE lmp_gather_atoms_concat_double(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
2022-10-04 11:49:17 +08:00
|
|
|
REAL(c_double) :: dnatoms
|
2022-10-03 09:32:42 +08:00
|
|
|
CHARACTER(LEN=100) :: error_msg
|
2022-10-03 04:28:10 +08:00
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_atoms_concat requires "count" to be 1 or 3 &
|
|
|
|
&[Fortran/gather_atoms_concat]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
2022-10-03 04:28:10 +08:00
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather_atoms_concat with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather_atoms_concat]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-10-03 04:28:10 +08:00
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_atoms_concat(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_atoms_concat_double
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_atoms_subset (for integers)
|
|
|
|
SUBROUTINE lmp_gather_atoms_subset_int(self, name, count, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET, INTENT(IN) :: ids
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int) :: ndata
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_atoms_subset requires "count" to be 1 or 3 &
|
2022-12-03 07:19:42 +08:00
|
|
|
&[Fortran/gather_atoms_subset]')
|
2022-10-03 04:28:10 +08:00
|
|
|
END IF
|
|
|
|
|
|
|
|
ndata = SIZE(ids, KIND=c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-10-03 04:28:10 +08:00
|
|
|
ALLOCATE(data(ndata*count))
|
|
|
|
data = -1_c_int
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Cids = C_LOC(ids(1))
|
|
|
|
CALL lammps_gather_atoms_subset(self%handle, Cname, Ctype, count, &
|
|
|
|
ndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_atoms_subset_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_atoms_subset (for doubles)
|
|
|
|
SUBROUTINE lmp_gather_atoms_subset_double(self, name, count, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET, INTENT(IN) :: ids
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int) :: ndata
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_atoms_subset requires "count" to be 1 or 3 &
|
2022-12-03 07:19:42 +08:00
|
|
|
&[Fortran/gather_atoms_subset]')
|
2022-10-03 04:28:10 +08:00
|
|
|
END IF
|
|
|
|
|
|
|
|
ndata = SIZE(ids, KIND=c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-10-03 04:28:10 +08:00
|
|
|
ALLOCATE(data(ndata*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Cids = C_LOC(ids(1))
|
|
|
|
CALL lammps_gather_atoms_subset(self%handle, Cname, Ctype, count, &
|
|
|
|
ndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_atoms_subset_double
|
|
|
|
|
2022-10-03 09:32:42 +08:00
|
|
|
! equivalent function to lammps_scatter_atoms (for integers)
|
|
|
|
SUBROUTINE lmp_scatter_atoms_int(self, name, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int) :: natoms, Ccount
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
TYPE(c_ptr) :: Cname, Cdata
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function scatter_atoms with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/scatter_atoms]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Ccount = SIZE(data) / natoms
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'lammps_scatter_atoms requires either 1 or 3 data per atom')
|
|
|
|
END IF
|
|
|
|
CALL lammps_scatter_atoms(self%handle, Cname, Ctype, Ccount, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_atoms_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_scatter_atoms (for doubles)
|
|
|
|
SUBROUTINE lmp_scatter_atoms_double(self, name, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
REAL(c_double), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int) :: natoms, Ccount
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
TYPE(c_ptr) :: Cname, Cdata
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function scatter_atoms with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/scatter_atoms]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Ccount = SIZE(data) / natoms
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'scatter_atoms requires either 1 or 3 data per atom &
|
|
|
|
&[Fortran/scatter_atoms]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_scatter_atoms(self%handle, Cname, Ctype, Ccount, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_atoms_double
|
|
|
|
|
2022-10-19 22:56:54 +08:00
|
|
|
! equivalent function to lammps_scatter_atoms_subset (for integers)
|
2022-10-03 09:32:42 +08:00
|
|
|
SUBROUTINE lmp_scatter_atoms_subset_int(self, name, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: ids
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
INTEGER(c_int) :: Cndata, Ccount
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
|
|
|
|
Cndata = SIZE(ids, KIND=c_int)
|
|
|
|
Ccount = SIZE(data, KIND=c_int) / Cndata
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2022-12-03 07:19:42 +08:00
|
|
|
'scatter_atoms_subset requires either 1 or 3 data per atom &
|
|
|
|
&[Fortran/scatter_atoms_subset]')
|
2022-10-03 09:32:42 +08:00
|
|
|
END IF
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
2022-10-04 06:44:09 +08:00
|
|
|
Cids = C_LOC(ids(1))
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lammps_scatter_atoms_subset(self%handle, Cname, Ctype, Ccount, &
|
|
|
|
Cndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_atoms_subset_int
|
|
|
|
|
2022-10-19 22:56:54 +08:00
|
|
|
! equivalent function to lammps_scatter_atoms_subset (for doubles)
|
2022-10-03 09:32:42 +08:00
|
|
|
SUBROUTINE lmp_scatter_atoms_subset_double(self, name, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: ids
|
|
|
|
REAL(c_double), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
INTEGER(c_int) :: Cndata, Ccount
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
|
|
|
|
Cndata = SIZE(ids, KIND=c_int)
|
|
|
|
Ccount = SIZE(data, KIND=c_int) / Cndata
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2022-12-03 07:19:42 +08:00
|
|
|
'scatter_atoms_subset requires either 1 or 3 data per atom &
|
|
|
|
&[Fortran/scatter_atoms_subset]')
|
2022-10-03 09:32:42 +08:00
|
|
|
END IF
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
2022-10-04 06:44:09 +08:00
|
|
|
Cids = C_LOC(ids(1))
|
2022-10-03 09:32:42 +08:00
|
|
|
CALL lammps_scatter_atoms_subset(self%handle, Cname, Ctype, Ccount, &
|
|
|
|
Cndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_atoms_subset_double
|
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
! equivalent function to lammps_gather_bonds (LAMMPS_SMALLSMALL or SMALLBIG)
|
|
|
|
SUBROUTINE lmp_gather_bonds_small(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
2022-10-28 13:51:04 +08:00
|
|
|
INTEGER(c_int), POINTER :: nbonds_small
|
|
|
|
INTEGER(c_int64_t), POINTER :: nbonds_big
|
2022-10-24 14:00:48 +08:00
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
2022-11-25 11:07:46 +08:00
|
|
|
IF (SIZE_TAGINT /= 4_c_int) THEN
|
2022-10-24 14:00:48 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2022-12-03 07:19:42 +08:00
|
|
|
'Incompatible integer kind in gather_bonds [Fortran/gather_bonds]')
|
2022-10-24 14:00:48 +08:00
|
|
|
END IF
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
2022-11-25 11:07:46 +08:00
|
|
|
IF (SIZE_BIGINT == 4_c_int) THEN
|
2022-10-28 13:51:04 +08:00
|
|
|
nbonds_small = lmp_extract_global(self, 'nbonds')
|
|
|
|
ALLOCATE(data(3*nbonds_small))
|
|
|
|
ELSE
|
|
|
|
nbonds_big = lmp_extract_global(self, 'nbonds')
|
|
|
|
ALLOCATE(data(3*nbonds_big))
|
|
|
|
END IF
|
2022-10-24 14:00:48 +08:00
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_bonds(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_bonds_small
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_bonds (LAMMPS_BIGBIG)
|
|
|
|
SUBROUTINE lmp_gather_bonds_big(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int64_t), POINTER :: nbonds
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
2022-11-25 11:07:46 +08:00
|
|
|
IF (SIZE_TAGINT /= 8_c_int) THEN
|
2022-10-24 14:00:48 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
2022-12-03 07:19:42 +08:00
|
|
|
'Incompatible integer kind in gather_bonds [Fortran/gather_bonds]')
|
2022-10-24 14:00:48 +08:00
|
|
|
END IF
|
|
|
|
nbonds = lmp_extract_global(self, 'nbonds')
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(3*nbonds))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_bonds(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_bonds_big
|
|
|
|
|
2023-01-26 18:22:08 +08:00
|
|
|
! equivalent function to lammps_gather_angles (LAMMPS_SMALLSMALL or SMALLBIG)
|
|
|
|
SUBROUTINE lmp_gather_angles_small(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int), POINTER :: nangles_small
|
|
|
|
INTEGER(c_int64_t), POINTER :: nangles_big
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT /= 4_c_int) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Incompatible integer kind in gather_angles [Fortran/gather_angles]')
|
|
|
|
END IF
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
IF (SIZE_BIGINT == 4_c_int) THEN
|
|
|
|
nangles_small = lmp_extract_global(self, 'nangles')
|
|
|
|
ALLOCATE(data(4*nangles_small))
|
|
|
|
ELSE
|
|
|
|
nangles_big = lmp_extract_global(self, 'nangles')
|
|
|
|
ALLOCATE(data(4*nangles_big))
|
|
|
|
END IF
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_angles(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_angles_small
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_angles (LAMMPS_BIGBIG)
|
|
|
|
SUBROUTINE lmp_gather_angles_big(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int64_t), POINTER :: nangles
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT /= 8_c_int) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Incompatible integer kind in gather_angles [Fortran/gather_angles]')
|
|
|
|
END IF
|
|
|
|
nangles = lmp_extract_global(self, 'nangles')
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(4*nangles))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_angles(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_angles_big
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_dihedrals (LAMMPS_SMALLSMALL or SMALLBIG)
|
|
|
|
SUBROUTINE lmp_gather_dihedrals_small(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int), POINTER :: ndihedrals_small
|
|
|
|
INTEGER(c_int64_t), POINTER :: ndihedrals_big
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT /= 4_c_int) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Incompatible integer kind in gather_dihedrals [Fortran/gather_dihedrals]')
|
|
|
|
END IF
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
IF (SIZE_BIGINT == 4_c_int) THEN
|
|
|
|
ndihedrals_small = lmp_extract_global(self, 'ndihedrals')
|
|
|
|
ALLOCATE(data(5*ndihedrals_small))
|
|
|
|
ELSE
|
|
|
|
ndihedrals_big = lmp_extract_global(self, 'ndihedrals')
|
|
|
|
ALLOCATE(data(5*ndihedrals_big))
|
|
|
|
END IF
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_dihedrals(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_dihedrals_small
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_dihedrals (LAMMPS_BIGBIG)
|
|
|
|
SUBROUTINE lmp_gather_dihedrals_big(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int64_t), POINTER :: ndihedrals
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT /= 8_c_int) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Incompatible integer kind in gather_dihedrals [Fortran/gather_dihedrals]')
|
|
|
|
END IF
|
|
|
|
ndihedrals = lmp_extract_global(self, 'ndihedrals')
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(5*ndihedrals))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_dihedrals(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_dihedrals_big
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_impropers (LAMMPS_SMALLSMALL or SMALLBIG)
|
|
|
|
SUBROUTINE lmp_gather_impropers_small(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int), POINTER :: nimpropers_small
|
|
|
|
INTEGER(c_int64_t), POINTER :: nimpropers_big
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT /= 4_c_int) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Incompatible integer kind in gather_impropers [Fortran/gather_impropers]')
|
|
|
|
END IF
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
IF (SIZE_BIGINT == 4_c_int) THEN
|
|
|
|
nimpropers_small = lmp_extract_global(self, 'nimpropers')
|
|
|
|
ALLOCATE(data(5*nimpropers_small))
|
|
|
|
ELSE
|
|
|
|
nimpropers_big = lmp_extract_global(self, 'nimpropers')
|
|
|
|
ALLOCATE(data(5*nimpropers_big))
|
|
|
|
END IF
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_impropers(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_impropers_small
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_impropers (LAMMPS_BIGBIG)
|
|
|
|
SUBROUTINE lmp_gather_impropers_big(self, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int64_t), POINTER :: nimpropers
|
|
|
|
TYPE(c_ptr) :: Cdata
|
|
|
|
|
|
|
|
IF (SIZE_TAGINT /= 8_c_int) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Incompatible integer kind in gather_impropers [Fortran/gather_impropers]')
|
|
|
|
END IF
|
|
|
|
nimpropers = lmp_extract_global(self, 'nimpropers')
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(5*nimpropers))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_impropers(self%handle, Cdata)
|
|
|
|
END SUBROUTINE lmp_gather_impropers_big
|
|
|
|
|
2022-12-03 07:19:42 +08:00
|
|
|
! equivalent function to lammps_gather (for int data)
|
|
|
|
SUBROUTINE lmp_gather_int(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather requires "count" to be 1 or 3 [Fortran/gather]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_atoms (for doubles)
|
|
|
|
SUBROUTINE lmp_gather_double(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather requires "count" to be 1 or 3 [Fortran/gather]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_double
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_concat (for ints)
|
|
|
|
SUBROUTINE lmp_gather_concat_int(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_concat requires "count" to be 1 or 3 [Fortran/gather_concat]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather_concat with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather_concat]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_concat(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_concat_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_concat (for doubles)
|
|
|
|
SUBROUTINE lmp_gather_concat_double(self, name, count, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname
|
|
|
|
INTEGER(c_int) :: natoms
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_concat requires "count" to be 1 or 3 [Fortran/gather_concat]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function gather_concat with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/gather_concat]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(natoms*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
CALL lammps_gather_concat(self%handle, Cname, Ctype, count, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_concat_double
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_subset (for integers)
|
|
|
|
SUBROUTINE lmp_gather_subset_int(self, name, count, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET, INTENT(IN) :: ids
|
|
|
|
INTEGER(c_int), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int) :: ndata
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
|
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_subset requires "count" to be 1 or 3 [Fortran/gather_subset]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
ndata = SIZE(ids, KIND=c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(ndata*count))
|
|
|
|
data = -1_c_int
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Cids = C_LOC(ids(1))
|
|
|
|
CALL lammps_gather_subset(self%handle, Cname, Ctype, count, &
|
|
|
|
ndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_subset_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_gather_subset (for doubles)
|
|
|
|
SUBROUTINE lmp_gather_subset_double(self, name, count, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), INTENT(IN) :: count
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET, INTENT(IN) :: ids
|
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, TARGET, INTENT(OUT) :: data
|
|
|
|
INTEGER(c_int) :: ndata
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
|
|
|
|
IF (count /= 1 .AND. count /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'gather_subset requires "count" to be 1 or 3 [Fortran/gather_subset]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
ndata = SIZE(ids, KIND=c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
IF (ALLOCATED(data)) DEALLOCATE(data)
|
|
|
|
ALLOCATE(data(ndata*count))
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Cids = C_LOC(ids(1))
|
|
|
|
CALL lammps_gather_subset(self%handle, Cname, Ctype, count, &
|
|
|
|
ndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_gather_subset_double
|
|
|
|
|
|
|
|
! equivalent function to lammps_scatter (for integers)
|
|
|
|
SUBROUTINE lmp_scatter_int(self, name, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int) :: natoms, Ccount
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
TYPE(c_ptr) :: Cname, Cdata
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function scatter with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/scatter]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Ccount = SIZE(data) / natoms
|
|
|
|
|
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'lammps_scatter requires either 1 or 3 data per atom')
|
|
|
|
END IF
|
|
|
|
CALL lammps_scatter(self%handle, Cname, Ctype, Ccount, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_scatter (for doubles)
|
2022-12-04 06:03:29 +08:00
|
|
|
SUBROUTINE lmp_scatter_double(self, name, data)
|
2022-12-03 07:19:42 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
REAL(c_double), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int) :: natoms, Ccount
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
TYPE(c_ptr) :: Cname, Cdata
|
|
|
|
REAL(c_double) :: dnatoms
|
|
|
|
CHARACTER(LEN=100) :: error_msg
|
|
|
|
|
|
|
|
dnatoms = lmp_get_natoms(self)
|
|
|
|
IF (dnatoms > HUGE(1_c_int)) THEN
|
|
|
|
WRITE(error_msg,'(A,1X,I0,1X,A)') &
|
|
|
|
'Cannot use library function scatter with more than', &
|
|
|
|
HUGE(0_c_int), 'atoms [Fortran/scatter]'
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, error_msg)
|
|
|
|
END IF
|
|
|
|
natoms = NINT(dnatoms, c_int)
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Ccount = SIZE(data) / natoms
|
|
|
|
|
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'scatter requires either 1 or 3 data per atom [Fortran/scatter]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_scatter(self%handle, Cname, Ctype, Ccount, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_double
|
|
|
|
|
|
|
|
! equivalent function to lammps_scatter_subset (for integers)
|
|
|
|
SUBROUTINE lmp_scatter_subset_int(self, name, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: ids
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 0_c_int
|
|
|
|
INTEGER(c_int) :: Cndata, Ccount
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
|
|
|
|
Cndata = SIZE(ids, KIND=c_int)
|
|
|
|
Ccount = SIZE(data, KIND=c_int) / Cndata
|
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'scatter_subset requires either 1 or 3 data per atom &
|
|
|
|
&[Fortran/scatter_subset]')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Cids = C_LOC(ids(1))
|
|
|
|
CALL lammps_scatter_subset(self%handle, Cname, Ctype, Ccount, &
|
|
|
|
Cndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_subset_int
|
|
|
|
|
|
|
|
! equivalent function to lammps_scatter_subset (for doubles)
|
|
|
|
SUBROUTINE lmp_scatter_subset_double(self, name, ids, data)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: ids
|
|
|
|
REAL(c_double), DIMENSION(:), TARGET :: data
|
|
|
|
INTEGER(c_int), PARAMETER :: Ctype = 1_c_int
|
|
|
|
INTEGER(c_int) :: Cndata, Ccount
|
|
|
|
TYPE(c_ptr) :: Cdata, Cname, Cids
|
|
|
|
|
|
|
|
Cndata = SIZE(ids, KIND=c_int)
|
|
|
|
Ccount = SIZE(data, KIND=c_int) / Cndata
|
|
|
|
IF (Ccount /= 1 .AND. Ccount /= 3) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'scatter_subset requires either 1 or 3 data per atom')
|
|
|
|
END IF
|
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
Cdata = C_LOC(data(1))
|
|
|
|
Cids = C_LOC(ids(1))
|
|
|
|
CALL lammps_scatter_subset(self%handle, Cname, Ctype, Ccount, &
|
|
|
|
Cndata, Cids, Cdata)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END SUBROUTINE lmp_scatter_subset_double
|
|
|
|
|
2022-10-20 07:40:12 +08:00
|
|
|
! equivalent function to lammps_create_atoms (int ids or id absent)
|
2022-10-19 22:56:54 +08:00
|
|
|
SUBROUTINE lmp_create_atoms_int(self, id, type, x, v, image, bexpand)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-10-20 07:40:12 +08:00
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET, OPTIONAL :: id, image
|
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET, OPTIONAL :: type
|
|
|
|
REAL(c_double), DIMENSION(:), TARGET, OPTIONAL :: x, v
|
|
|
|
LOGICAL, OPTIONAL :: bexpand
|
2022-10-19 22:56:54 +08:00
|
|
|
INTEGER(c_int) :: n, Cbexpand
|
|
|
|
TYPE(c_ptr) :: Cid, Ctype, Cx, Cv, Cimage
|
|
|
|
INTEGER(c_int) :: tagint_size, atoms_created
|
|
|
|
|
2024-09-07 02:00:27 +08:00
|
|
|
Ctype = c_null_ptr
|
|
|
|
Cx = c_null_ptr
|
2022-10-20 07:40:12 +08:00
|
|
|
! type is actually NOT optional, but we can't make id optional without it,
|
|
|
|
! so we check at run-time
|
|
|
|
IF (.NOT. PRESENT(type)) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'the "type" argument to create_atoms is required&
|
|
|
|
& [Fortran/create_atoms]')
|
|
|
|
END IF
|
|
|
|
|
2022-10-19 22:56:54 +08:00
|
|
|
tagint_size = lmp_extract_setting(self, 'tagint')
|
2022-10-20 07:40:12 +08:00
|
|
|
IF (tagint_size /= 4_c_int .AND. (PRESENT(id) .OR. PRESENT(image))) THEN
|
2022-10-19 22:56:54 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Unable to create_atoms; your id/image array types are incompatible&
|
|
|
|
& with LAMMPS_SMALLBIG and LAMMPS_SMALLSMALL [Fortran/create_atoms]')
|
|
|
|
END IF
|
2022-10-20 07:40:12 +08:00
|
|
|
n = SIZE(type, KIND=c_int)
|
|
|
|
IF (PRESENT(bexpand)) THEN
|
|
|
|
IF (bexpand) THEN
|
|
|
|
Cbexpand = 1_c_int
|
|
|
|
ELSE
|
|
|
|
Cbexpand = 0_c_int
|
|
|
|
END IF
|
2022-10-19 22:56:54 +08:00
|
|
|
ELSE
|
2022-10-20 07:40:12 +08:00
|
|
|
Cbexpand = 0_c_int
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(id)) THEN
|
|
|
|
Cid = C_LOC(id(1))
|
|
|
|
ELSE
|
|
|
|
Cid = c_null_ptr
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(type)) THEN
|
|
|
|
Ctype = C_LOC(type(1))
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(image)) THEN
|
|
|
|
Cimage = C_LOC(image(1))
|
|
|
|
ELSE
|
|
|
|
Cimage = c_null_ptr
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(x)) THEN
|
|
|
|
Cx = C_LOC(x(1))
|
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'the argument "x" to create_atoms is required')
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(v)) THEN
|
|
|
|
Cv = C_LOC(v(1))
|
|
|
|
ELSE
|
|
|
|
Cv = c_null_ptr
|
2022-10-19 22:56:54 +08:00
|
|
|
END IF
|
|
|
|
atoms_created = lammps_create_atoms(self%handle, n, Cid, Ctype, Cx, Cv, &
|
|
|
|
Cimage, Cbexpand)
|
|
|
|
IF ( atoms_created < 0_c_int ) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'error when trying to create atoms [Fortran/create_atoms]')
|
|
|
|
ELSE IF ( atoms_created /= n ) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'atoms created /= atoms asked to create [Fortran/create_atoms]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_create_atoms_int
|
|
|
|
|
2022-10-20 07:40:12 +08:00
|
|
|
! equivalent function to lammps_create_atoms (long int ids and images)
|
2022-10-19 22:56:54 +08:00
|
|
|
SUBROUTINE lmp_create_atoms_bigbig(self, id, type, x, v, image, bexpand)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-10-20 07:40:12 +08:00
|
|
|
INTEGER(c_int64_t), DIMENSION(:), TARGET :: id
|
2022-10-19 22:56:54 +08:00
|
|
|
INTEGER(c_int), DIMENSION(:), TARGET :: type
|
2022-10-20 07:40:12 +08:00
|
|
|
REAL(c_double), DIMENSION(:), TARGET :: x
|
|
|
|
REAL(c_double), DIMENSION(:), OPTIONAL, TARGET :: v
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), OPTIONAL, TARGET :: image
|
|
|
|
LOGICAL, OPTIONAL :: bexpand
|
2022-10-19 22:56:54 +08:00
|
|
|
INTEGER(c_int) :: n, Cbexpand
|
|
|
|
TYPE(c_ptr) :: Cid, Ctype, Cx, Cv, Cimage
|
|
|
|
INTEGER(c_int) :: tagint_size, atoms_created
|
|
|
|
|
|
|
|
tagint_size = lmp_extract_setting(self, 'tagint')
|
|
|
|
IF ( tagint_size /= 8_c_int ) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Unable to create_atoms; your id/image array types are incompatible&
|
|
|
|
& with LAMMPS_BIGBIG')
|
|
|
|
END IF
|
2022-10-20 07:40:12 +08:00
|
|
|
n = SIZE(type, KIND=c_int)
|
|
|
|
IF (PRESENT(bexpand)) THEN
|
|
|
|
IF (bexpand) THEN
|
|
|
|
Cbexpand = 1_c_int
|
|
|
|
ELSE
|
|
|
|
Cbexpand = 0_c_int
|
|
|
|
END IF
|
2022-10-19 22:56:54 +08:00
|
|
|
ELSE
|
2022-10-20 07:40:12 +08:00
|
|
|
Cbexpand = 0_c_int
|
2022-10-19 22:56:54 +08:00
|
|
|
END IF
|
|
|
|
Cid = C_LOC(id(1))
|
|
|
|
Ctype = C_LOC(type(1))
|
2022-10-20 07:40:12 +08:00
|
|
|
IF (PRESENT(image)) THEN
|
|
|
|
Cimage = C_LOC(image(1))
|
|
|
|
ELSE
|
|
|
|
Cimage = c_null_ptr
|
|
|
|
END IF
|
2022-10-19 22:56:54 +08:00
|
|
|
Cx = C_LOC(x(1))
|
2022-10-20 07:40:12 +08:00
|
|
|
IF (PRESENT(v)) THEN
|
|
|
|
Cv = C_LOC(v(1))
|
|
|
|
ELSE
|
|
|
|
Cv = c_null_ptr
|
|
|
|
END IF
|
2022-10-19 22:56:54 +08:00
|
|
|
atoms_created = lammps_create_atoms(self%handle, n, Cid, Ctype, Cx, Cv, &
|
|
|
|
Cimage, Cbexpand)
|
|
|
|
IF ( atoms_created < 0_c_int ) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'error when trying to create atoms [Fortran/create_atoms]')
|
|
|
|
ELSE IF ( atoms_created /= n ) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'atoms created /= atoms asked to create [Fortran/create_atoms]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_create_atoms_bigbig
|
|
|
|
|
2022-10-24 14:00:48 +08:00
|
|
|
! equivalent function to lammps_find_pair_neighlist
|
|
|
|
INTEGER(c_int) FUNCTION lmp_find_pair_neighlist(self, style, exact, nsub, &
|
|
|
|
reqid)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: style
|
2022-10-29 09:53:46 +08:00
|
|
|
LOGICAL, INTENT(IN), OPTIONAL :: exact
|
|
|
|
INTEGER(c_int), INTENT(IN), OPTIONAL :: nsub, reqid
|
2022-10-24 14:00:48 +08:00
|
|
|
TYPE(c_ptr) :: Cstyle
|
2022-10-29 09:53:46 +08:00
|
|
|
INTEGER(c_int) :: Cexact, Cnsub, Creqid
|
2022-10-24 14:00:48 +08:00
|
|
|
|
2022-10-29 09:53:46 +08:00
|
|
|
Cexact = 0_c_int
|
|
|
|
IF (PRESENT(exact)) THEN
|
|
|
|
IF (exact) THEN
|
|
|
|
Cexact = 1_c_int
|
|
|
|
END IF
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(nsub)) THEN
|
|
|
|
Cnsub = nsub
|
|
|
|
ELSE
|
|
|
|
Cnsub = 0_c_int
|
|
|
|
END IF
|
|
|
|
IF (PRESENT(reqid)) THEN
|
|
|
|
Creqid = reqid
|
2022-10-24 14:00:48 +08:00
|
|
|
ELSE
|
2022-10-29 09:53:46 +08:00
|
|
|
Creqid = 0_c_int
|
2022-10-24 14:00:48 +08:00
|
|
|
END IF
|
|
|
|
Cstyle = f2c_string(style)
|
|
|
|
lmp_find_pair_neighlist = lammps_find_pair_neighlist(self%handle, Cstyle, &
|
2022-10-29 09:53:46 +08:00
|
|
|
Cexact, Cnsub, Creqid)
|
2022-10-24 14:00:48 +08:00
|
|
|
IF (lmp_find_pair_neighlist < 0) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'unable to find pair neighbor list [Fortran/find_pair_neighlist]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_free(Cstyle)
|
|
|
|
END FUNCTION lmp_find_pair_neighlist
|
|
|
|
|
|
|
|
! equivalent function to lammps_find_fix_neighlist
|
|
|
|
INTEGER(c_int) FUNCTION lmp_find_fix_neighlist(self, id, reqid) RESULT(idx)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
2022-10-29 10:01:12 +08:00
|
|
|
INTEGER(c_int), INTENT(IN), OPTIONAL :: reqid
|
2022-10-24 14:00:48 +08:00
|
|
|
TYPE(c_ptr) :: Cid
|
2022-10-29 10:01:12 +08:00
|
|
|
INTEGER(c_int) :: Creqid
|
2022-10-24 14:00:48 +08:00
|
|
|
|
2022-10-29 10:01:12 +08:00
|
|
|
IF (PRESENT(reqid)) THEN
|
|
|
|
Creqid = reqid
|
|
|
|
ELSE
|
|
|
|
Creqid = 0_c_int
|
|
|
|
END IF
|
2022-10-24 14:00:48 +08:00
|
|
|
Cid = f2c_string(id)
|
2022-10-29 10:01:12 +08:00
|
|
|
idx = lammps_find_fix_neighlist(self%handle, Cid, Creqid)
|
2022-10-24 14:00:48 +08:00
|
|
|
IF (idx < 0) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'neighbor list not found [Fortran/find_fix_neighlist]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END FUNCTION lmp_find_fix_neighlist
|
|
|
|
|
|
|
|
! equivalent function to lammps_find_compute_neighlist
|
|
|
|
INTEGER(c_int) FUNCTION lmp_find_compute_neighlist(self, id, reqid) &
|
|
|
|
RESULT(idx)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
2022-10-29 10:01:12 +08:00
|
|
|
INTEGER(c_int), INTENT(IN), OPTIONAL :: reqid
|
2022-10-24 14:00:48 +08:00
|
|
|
TYPE(c_ptr) :: Cid
|
2022-10-29 10:01:12 +08:00
|
|
|
INTEGER(c_int) :: Creqid
|
2022-10-24 14:00:48 +08:00
|
|
|
|
2022-10-29 10:01:12 +08:00
|
|
|
IF (PRESENT(reqid)) THEN
|
|
|
|
Creqid = reqid
|
|
|
|
ELSE
|
|
|
|
Creqid = 0_c_int
|
|
|
|
END IF
|
2022-10-24 14:00:48 +08:00
|
|
|
Cid = f2c_string(id)
|
2022-10-29 10:01:12 +08:00
|
|
|
idx = lammps_find_compute_neighlist(self%handle, Cid, Creqid)
|
2022-10-24 14:00:48 +08:00
|
|
|
IF (idx < 0) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'neighbor list not found [Fortran/find_compute_neighlist]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END FUNCTION lmp_find_compute_neighlist
|
|
|
|
|
|
|
|
INTEGER(c_int) FUNCTION lmp_neighlist_num_elements(self, idx) RESULT(inum)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), INTENT(IN) :: idx
|
|
|
|
|
|
|
|
inum = lammps_neighlist_num_elements(self%handle, idx)
|
|
|
|
IF (inum < 0) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'neighbor list not found [Fortran/neighlist_num_elements]')
|
|
|
|
END IF
|
|
|
|
END FUNCTION lmp_neighlist_num_elements
|
|
|
|
|
|
|
|
SUBROUTINE lmp_neighlist_element_neighbors(self, idx, element, iatom, &
|
|
|
|
neighbors)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), INTENT(IN) :: idx, element
|
|
|
|
INTEGER(c_int), INTENT(OUT) :: iatom
|
|
|
|
INTEGER(c_int), DIMENSION(:), POINTER, INTENT(OUT) :: neighbors
|
|
|
|
INTEGER(c_int) :: numneigh
|
|
|
|
TYPE(c_ptr) :: Cneighbors
|
|
|
|
|
|
|
|
CALL lammps_neighlist_element_neighbors(self%handle, idx, element, iatom, &
|
|
|
|
numneigh, Cneighbors)
|
2022-12-04 11:39:14 +08:00
|
|
|
IF (C_ASSOCIATED(Cneighbors)) THEN
|
|
|
|
CALL C_F_POINTER(Cneighbors, neighbors, [numneigh])
|
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Pointer returned from lammps_neighlist_element_neighbors is NULL')
|
|
|
|
END IF
|
2022-10-24 14:00:48 +08:00
|
|
|
END SUBROUTINE lmp_neighlist_element_neighbors
|
|
|
|
|
2022-10-01 07:31:18 +08:00
|
|
|
! equivalent function to lammps_version
|
2022-08-10 10:18:33 +08:00
|
|
|
INTEGER FUNCTION lmp_version(self)
|
2022-09-24 05:23:51 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-08-10 10:18:33 +08:00
|
|
|
|
|
|
|
lmp_version = lammps_version(self%handle)
|
|
|
|
END FUNCTION lmp_version
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
! equivalent function to lammps_get_os_info
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE lmp_get_os_info(buffer)
|
2022-09-30 08:28:15 +08:00
|
|
|
CHARACTER(LEN=*) :: buffer
|
2022-10-28 03:12:26 +08:00
|
|
|
INTEGER(c_int) :: buf_size
|
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(LEN(buffer)+1), TARGET :: Cbuffer
|
2022-09-30 08:28:15 +08:00
|
|
|
TYPE(c_ptr) :: ptr
|
|
|
|
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-28 03:12:26 +08:00
|
|
|
buf_size = LEN(buffer, KIND=c_int) + 1_c_int
|
|
|
|
ptr = C_LOC(Cbuffer(1))
|
2022-10-26 08:51:08 +08:00
|
|
|
CALL lammps_get_os_info(ptr, INT(buf_size, KIND=c_int))
|
2022-10-28 13:51:04 +08:00
|
|
|
buffer = array2string(Cbuffer)
|
2022-09-30 08:28:15 +08:00
|
|
|
END SUBROUTINE lmp_get_os_info
|
|
|
|
|
|
|
|
! equivalent function to lammps_config_has_mpi_support
|
|
|
|
LOGICAL FUNCTION lmp_config_has_mpi_support()
|
|
|
|
INTEGER(c_int) :: has_mpi_support
|
|
|
|
|
|
|
|
has_mpi_support = lammps_config_has_mpi_support()
|
|
|
|
lmp_config_has_mpi_support = (has_mpi_support /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_mpi_support
|
|
|
|
|
|
|
|
! equivalent function to lammps_config_has_gzip_support
|
|
|
|
LOGICAL FUNCTION lmp_config_has_gzip_support()
|
|
|
|
INTEGER(c_int) :: has_gzip_support
|
|
|
|
|
|
|
|
has_gzip_support = lammps_config_has_gzip_support()
|
|
|
|
lmp_config_has_gzip_support = (has_gzip_support /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_gzip_support
|
|
|
|
|
|
|
|
! equivalent function to lammps_config_has_png_support
|
|
|
|
LOGICAL FUNCTION lmp_config_has_png_support()
|
2022-10-04 11:49:17 +08:00
|
|
|
INTEGER(c_int) :: has_png_support
|
2022-09-30 08:28:15 +08:00
|
|
|
|
|
|
|
has_png_support = lammps_config_has_png_support()
|
|
|
|
lmp_config_has_png_support = (has_png_support /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_png_support
|
|
|
|
|
|
|
|
! equivalent function to lammps_config_has_jpeg_support
|
|
|
|
LOGICAL FUNCTION lmp_config_has_jpeg_support()
|
|
|
|
INTEGER(c_int) :: has_jpeg_support
|
|
|
|
|
|
|
|
has_jpeg_support = lammps_config_has_jpeg_support()
|
|
|
|
lmp_config_has_jpeg_support = (has_jpeg_support /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_jpeg_support
|
|
|
|
|
|
|
|
! equivalent function to lammps_config_has_ffmpeg_support
|
|
|
|
LOGICAL FUNCTION lmp_config_has_ffmpeg_support()
|
|
|
|
INTEGER(c_int) :: has_ffmpeg_support
|
|
|
|
|
|
|
|
has_ffmpeg_support = lammps_config_has_ffmpeg_support()
|
|
|
|
lmp_config_has_ffmpeg_support = (has_ffmpeg_support /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_ffmpeg_support
|
|
|
|
|
2024-08-07 22:37:20 +08:00
|
|
|
! equivalent function to lammps_config_has_curl_support
|
|
|
|
LOGICAL FUNCTION lmp_config_has_curl_support()
|
|
|
|
INTEGER(c_int) :: has_curl_support
|
|
|
|
|
|
|
|
has_curl_support = lammps_config_has_curl_support()
|
|
|
|
lmp_config_has_curl_support = (has_curl_support /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_curl_support
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
! equivalent function to lammps_config_has_exceptions
|
|
|
|
LOGICAL FUNCTION lmp_config_has_exceptions()
|
|
|
|
INTEGER(c_int) :: has_exceptions
|
|
|
|
|
|
|
|
has_exceptions = lammps_config_has_exceptions()
|
|
|
|
lmp_config_has_exceptions = (has_exceptions /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_has_exceptions
|
|
|
|
|
|
|
|
! equivalent function to lammps_config_has_package
|
|
|
|
LOGICAL FUNCTION lmp_config_has_package(name)
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: name
|
2022-10-01 07:50:16 +08:00
|
|
|
INTEGER(c_int) :: has_package
|
|
|
|
TYPE(c_ptr) :: Cname
|
2022-09-30 08:28:15 +08:00
|
|
|
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
has_package = lammps_config_has_package(Cname)
|
|
|
|
lmp_config_has_package = (has_package /= 0_c_int)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
END FUNCTION lmp_config_has_package
|
|
|
|
|
|
|
|
! equivalent subroutine to lammps_config_package_name
|
2022-10-26 08:51:08 +08:00
|
|
|
SUBROUTINE lmp_config_package_name(self, idx, buffer)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-09-30 08:28:15 +08:00
|
|
|
INTEGER, INTENT(IN) :: idx
|
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: buffer
|
2022-10-28 03:12:26 +08:00
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(LEN(buffer)+1), TARGET :: Cbuffer
|
2022-09-30 08:28:15 +08:00
|
|
|
INTEGER(c_int) :: Cidx, Csuccess
|
|
|
|
TYPE(c_ptr) :: Cptr
|
|
|
|
|
|
|
|
Cidx = idx - 1
|
2022-10-28 03:12:26 +08:00
|
|
|
Cptr = C_LOC(Cbuffer(1))
|
2022-09-30 08:28:15 +08:00
|
|
|
Csuccess = lammps_config_package_name(Cidx, Cptr, LEN(buffer)+1)
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (Csuccess /= 0_c_int) THEN
|
2022-10-28 03:12:26 +08:00
|
|
|
buffer = array2string(Cbuffer)
|
2022-10-26 08:51:08 +08:00
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'failure of lammps_config_package_name [Fortran/config_package_name]')
|
2022-09-30 08:28:15 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_config_package_name
|
|
|
|
|
2022-10-28 13:51:04 +08:00
|
|
|
! equivalent function to Python routine lammps.installed_packages()
|
2022-10-26 08:51:08 +08:00
|
|
|
SUBROUTINE lmp_installed_packages(self, package, length)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-09-30 08:28:15 +08:00
|
|
|
CHARACTER(LEN=:), DIMENSION(:), ALLOCATABLE, INTENT(OUT) :: package
|
|
|
|
INTEGER, INTENT(IN), OPTIONAL :: length
|
|
|
|
INTEGER, PARAMETER :: MAX_BUFFER_LENGTH = 31
|
|
|
|
INTEGER :: i, npackage, buf_length
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(length)) THEN
|
2022-09-30 08:28:15 +08:00
|
|
|
buf_length = length
|
|
|
|
ELSE
|
|
|
|
buf_length = MAX_BUFFER_LENGTH
|
|
|
|
END IF
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (ALLOCATED(package)) DEALLOCATE(package)
|
2022-09-30 08:28:15 +08:00
|
|
|
npackage = lammps_config_package_count()
|
2022-10-04 11:49:17 +08:00
|
|
|
ALLOCATE(CHARACTER(LEN=MAX_BUFFER_LENGTH) :: package(npackage))
|
2022-09-30 08:28:15 +08:00
|
|
|
DO i=1, npackage
|
2022-10-26 08:51:08 +08:00
|
|
|
CALL lmp_config_package_name(self, i, package(i))
|
2022-09-30 08:28:15 +08:00
|
|
|
END DO
|
|
|
|
END SUBROUTINE lmp_installed_packages
|
|
|
|
|
2022-10-21 09:24:59 +08:00
|
|
|
! equivalent function to lammps_config_accelerator
|
|
|
|
LOGICAL FUNCTION lmp_config_accelerator(package, category, setting)
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: package, category, setting
|
|
|
|
TYPE(c_ptr) :: Cpackage, Ccategory, Csetting
|
|
|
|
INTEGER(c_int) :: is_configured
|
|
|
|
|
|
|
|
Cpackage = f2c_string(package)
|
|
|
|
Ccategory = f2c_string(category)
|
|
|
|
Csetting = f2c_string(setting)
|
|
|
|
is_configured = lammps_config_accelerator(Cpackage, Ccategory, Csetting)
|
|
|
|
CALL lammps_free(Cpackage)
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
CALL lammps_free(Csetting)
|
|
|
|
lmp_config_accelerator = (is_configured /= 0_c_int)
|
|
|
|
END FUNCTION lmp_config_accelerator
|
|
|
|
|
|
|
|
! equivalent function to lammps_has_gpu_device
|
|
|
|
LOGICAL FUNCTION lmp_has_gpu_device()
|
|
|
|
lmp_has_gpu_device = (lammps_has_gpu_device() /= 0_c_int)
|
|
|
|
END FUNCTION lmp_has_gpu_device
|
|
|
|
|
|
|
|
! equivalent subroutine to lammps_get_gpu_device_info
|
|
|
|
SUBROUTINE lmp_get_gpu_device_info(buffer)
|
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: buffer
|
2022-10-31 05:42:10 +08:00
|
|
|
INTEGER(c_int) :: buf_size
|
2022-10-28 03:12:26 +08:00
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(LEN(buffer)+1), TARGET :: Cbuffer
|
2022-10-21 09:24:59 +08:00
|
|
|
TYPE(c_ptr) :: Cptr
|
|
|
|
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-21 09:24:59 +08:00
|
|
|
buf_size = LEN(buffer) + 1
|
2022-10-28 03:12:26 +08:00
|
|
|
Cptr = C_LOC(Cbuffer)
|
2022-10-21 09:24:59 +08:00
|
|
|
CALL lammps_get_gpu_device_info(Cptr, buf_size)
|
2022-10-28 03:12:26 +08:00
|
|
|
buffer = array2string(Cbuffer)
|
2022-10-21 09:24:59 +08:00
|
|
|
END SUBROUTINE lmp_get_gpu_device_info
|
|
|
|
|
|
|
|
! equivalent function to lammps_has_style
|
|
|
|
LOGICAL FUNCTION lmp_has_style(self, category, name)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: category, name
|
|
|
|
TYPE(c_ptr) :: Ccategory, Cname
|
|
|
|
INTEGER(c_int) :: has_style
|
|
|
|
|
|
|
|
Ccategory = f2c_string(category)
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
has_style = lammps_has_style(self%handle, Ccategory, Cname)
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
lmp_has_style = (has_style /= 0_c_int)
|
|
|
|
END FUNCTION
|
|
|
|
|
|
|
|
! equivalent function to lammps_style_count
|
|
|
|
INTEGER(c_int) FUNCTION lmp_style_count(self, category)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: category
|
|
|
|
TYPE(c_ptr) :: Ccategory
|
|
|
|
|
|
|
|
Ccategory = f2c_string(category)
|
|
|
|
lmp_style_count = lammps_style_count(self%handle, Ccategory)
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
END FUNCTION lmp_style_count
|
|
|
|
|
|
|
|
! equivalent function to lammps_style_name
|
|
|
|
SUBROUTINE lmp_style_name(self, category, idx, buffer)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: category
|
|
|
|
INTEGER(c_int), INTENT(IN) :: idx
|
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: buffer
|
|
|
|
INTEGER(c_int) :: buf_size, success
|
2022-10-28 03:12:26 +08:00
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(LEN(buffer)+1), TARGET :: Cbuffer
|
|
|
|
TYPE(c_ptr) :: Ccategory, Cptr
|
2022-10-21 09:24:59 +08:00
|
|
|
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-28 03:12:26 +08:00
|
|
|
buf_size = LEN(buffer, KIND=c_int) + 1_c_int
|
2022-10-21 09:24:59 +08:00
|
|
|
Ccategory = f2c_string(category)
|
2022-10-28 03:12:26 +08:00
|
|
|
Cptr = C_LOC(Cbuffer)
|
2022-10-29 08:09:31 +08:00
|
|
|
success = lammps_style_name(self%handle, Ccategory, idx - 1, Cptr, buf_size)
|
2022-10-21 09:24:59 +08:00
|
|
|
IF (success == 1_c_int) THEN
|
2022-10-28 03:12:26 +08:00
|
|
|
buffer = array2string(Cbuffer)
|
2022-10-21 09:24:59 +08:00
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'idx value not in range [Fortran/style_name]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
END SUBROUTINE lmp_style_name
|
|
|
|
|
2022-10-26 08:51:08 +08:00
|
|
|
! equivalent function to lammps_has_id
|
|
|
|
LOGICAL FUNCTION lmp_has_id(self, category, name)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: category, name
|
|
|
|
TYPE(c_ptr) :: Ccategory, Cname
|
|
|
|
INTEGER(c_int) :: has_id
|
|
|
|
|
|
|
|
Ccategory = f2c_string(category)
|
|
|
|
Cname = f2c_string(name)
|
|
|
|
has_id = lammps_has_id(self%handle, Ccategory, Cname)
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
CALL lammps_free(Cname)
|
|
|
|
lmp_has_id = (has_id /= 0_c_int)
|
|
|
|
END FUNCTION lmp_has_id
|
|
|
|
|
|
|
|
! equivalent function to lammps_id_count
|
|
|
|
INTEGER(c_int) FUNCTION lmp_id_count(self, category)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: category
|
|
|
|
TYPE(c_ptr) :: Ccategory
|
|
|
|
|
|
|
|
Ccategory = f2c_string(category)
|
|
|
|
lmp_id_count = lammps_id_count(self%handle, Ccategory)
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
END FUNCTION lmp_id_count
|
|
|
|
|
|
|
|
! equivalent function to lammps_id_name
|
|
|
|
SUBROUTINE lmp_id_name(self, category, idx, buffer)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: category
|
|
|
|
INTEGER(c_int), INTENT(IN) :: idx
|
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: buffer
|
|
|
|
INTEGER(c_int) :: success
|
|
|
|
INTEGER(c_int) :: buf_size
|
2022-10-28 03:12:26 +08:00
|
|
|
TYPE(c_ptr) :: Ccategory, Cptr
|
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(LEN(buffer)+1), TARGET :: Cbuffer
|
2022-10-26 08:51:08 +08:00
|
|
|
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-26 08:51:08 +08:00
|
|
|
Ccategory = f2c_string(category)
|
|
|
|
buf_size = LEN(buffer, KIND=c_int)
|
2022-10-28 03:12:26 +08:00
|
|
|
Cptr = C_LOC(Cbuffer(1))
|
2022-10-29 08:09:31 +08:00
|
|
|
success = lammps_id_name(self%handle, Ccategory, idx - 1, Cptr, buf_size)
|
2022-10-26 08:51:08 +08:00
|
|
|
IF (success /= 0) THEN
|
2022-10-28 03:12:26 +08:00
|
|
|
buffer = array2string(Cbuffer)
|
2022-10-26 08:51:08 +08:00
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'lammps_id_name failed [Fortran/id_name]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_free(Ccategory)
|
|
|
|
END SUBROUTINE lmp_id_name
|
|
|
|
|
2022-10-26 13:02:54 +08:00
|
|
|
! equivalent function to lammps_plugin_name
|
|
|
|
SUBROUTINE lmp_plugin_name(self, idx, stylebuf, namebuf)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), INTENT(IN) :: idx
|
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: stylebuf, namebuf
|
|
|
|
INTEGER(c_int) :: buf_size, success
|
|
|
|
TYPE(c_ptr) :: Cstylebuf, Cnamebuf
|
|
|
|
|
|
|
|
buf_size = MIN(LEN(stylebuf, KIND=c_int), LEN(namebuf, KIND=c_int))
|
|
|
|
Cstylebuf = lammps_malloc(INT(buf_size, KIND=c_size_t))
|
|
|
|
Cnamebuf = lammps_malloc(INT(buf_size, KIND=c_size_t))
|
2022-10-29 08:09:31 +08:00
|
|
|
success = lammps_plugin_name(idx - 1, Cstylebuf, Cnamebuf, buf_size)
|
2022-10-26 13:02:54 +08:00
|
|
|
IF (success /= 0_c_int) THEN
|
|
|
|
stylebuf = c2f_string(Cstylebuf)
|
|
|
|
namebuf = c2f_string(Cnamebuf)
|
|
|
|
ELSE
|
2022-12-23 12:23:32 +08:00
|
|
|
stylebuf = ' '
|
|
|
|
namebuf = ' '
|
2022-10-26 13:02:54 +08:00
|
|
|
CALL lmp_error(self, LMP_ERROR_WARNING + LMP_ERROR_WORLD, &
|
|
|
|
'call to lammps_plugin_name failed [Fortran/plugin_name]')
|
|
|
|
END IF
|
|
|
|
CALL lammps_free(Cstylebuf)
|
|
|
|
CALL lammps_free(Cnamebuf)
|
|
|
|
END SUBROUTINE lmp_plugin_name
|
|
|
|
|
2022-10-20 07:40:12 +08:00
|
|
|
! equivalent function to lammps_encode_image_flags
|
|
|
|
FUNCTION lmp_encode_image_flags(self, ix, iy, iz) RESULT (image)
|
|
|
|
CLASS(lammps), INTENT(IN), TARGET :: self
|
|
|
|
INTEGER(c_int), INTENT(IN) :: ix, iy, iz
|
|
|
|
TYPE(lammps_image_data) :: image
|
|
|
|
INTEGER(c_int) :: imageint_size
|
|
|
|
INTEGER(c_int) :: IMGMAX, IMGMASK, IMGBITS, IMG2BITS
|
|
|
|
INTEGER(c_int64_t) :: ibx, iby, ibz, BIMGMAX, BIMGMASK, BIMGBITS, BIMG2BITS
|
|
|
|
|
|
|
|
image%lammps_instance => self
|
|
|
|
IMGMASK = lmp_extract_setting(self, 'IMGMASK')
|
|
|
|
IMGMAX = lmp_extract_setting(self, 'IMGMAX')
|
|
|
|
IMGBITS = lmp_extract_setting(self, 'IMGBITS')
|
|
|
|
IMG2BITS = lmp_extract_setting(self, 'IMG2BITS')
|
|
|
|
imageint_size = lmp_extract_setting(self, 'imageint')
|
|
|
|
IF (imageint_size == 4_c_int) THEN
|
|
|
|
image%datatype = DATA_INT
|
|
|
|
image%i32 = IOR( IOR(IAND(ix + IMGMAX, IMGMASK), &
|
|
|
|
ISHFT(IAND(iy + IMGMAX, IMGMASK), IMGBITS)), &
|
|
|
|
ISHFT(IAND(iz + IMGMAX, IMGMASK), IMG2BITS) )
|
|
|
|
ELSE
|
|
|
|
image%datatype = DATA_INT64
|
|
|
|
ibx = ix
|
|
|
|
iby = iy
|
|
|
|
ibz = iz
|
|
|
|
BIMGMAX = IMGMAX
|
|
|
|
BIMGMASK = IMGMASK
|
|
|
|
BIMGBITS = IMGBITS
|
|
|
|
BIMG2BITS = IMG2BITS
|
|
|
|
image%i64 = IOR( IOR(IAND(ibx + BIMGMAX, BIMGMASK), &
|
|
|
|
ISHFT(IAND(iby + BIMGMAX, BIMGMASK), BIMGBITS)), &
|
|
|
|
ISHFT(IAND(ibz + BIMGMAX, BIMGMASK), BIMG2BITS) )
|
|
|
|
END IF
|
|
|
|
END FUNCTION lmp_encode_image_flags
|
|
|
|
|
|
|
|
! equivalent function to lammps_decode_image_flags
|
2022-10-20 13:27:57 +08:00
|
|
|
SUBROUTINE lmp_decode_image_flags(self, image, flags)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int), INTENT(IN) :: image
|
|
|
|
INTEGER(c_int), DIMENSION(3), TARGET, INTENT(OUT) :: flags
|
|
|
|
INTEGER(c_int) :: size_imageint
|
|
|
|
INTEGER(c_int) :: IMGMASK, IMGMAX, IMGBITS, IMG2BITS
|
|
|
|
|
|
|
|
size_imageint = lmp_extract_setting(self, 'imageint')
|
|
|
|
IF (size_imageint == 4_c_int) THEN
|
|
|
|
IMGMASK = lmp_extract_setting(self, 'IMGMASK')
|
|
|
|
IMGMAX = lmp_extract_setting(self, 'IMGMAX')
|
|
|
|
IMGBITS = lmp_extract_setting(self, 'IMGBITS')
|
|
|
|
IMG2BITS = lmp_extract_setting(self, 'IMG2BITS')
|
|
|
|
flags(1) = IAND(image, IMGMASK) - IMGMAX
|
|
|
|
flags(2) = IAND(ISHFT(image, -IMGBITS), IMGMASK) - IMGMAX
|
|
|
|
flags(3) = ISHFT(image, -IMG2BITS) - IMGMAX
|
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, 'Incorrect&
|
|
|
|
& integer kind passed as "image" [Fortran/decode_image_flags]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_decode_image_flags
|
|
|
|
|
|
|
|
! equivalent function to lammps_decode_image_flags if -DLAMMPS_BIGBIG is used
|
|
|
|
SUBROUTINE lmp_decode_image_flags_bigbig(self, image, flags)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
INTEGER(c_int64_t), INTENT(IN) :: image
|
|
|
|
INTEGER(c_int), DIMENSION(3), TARGET, INTENT(OUT) :: flags
|
|
|
|
INTEGER(c_int) :: size_imageint
|
|
|
|
INTEGER(c_int) :: IMGMASK, IMGMAX, IMGBITS, IMG2BITS
|
|
|
|
INTEGER(c_int64_t) :: BIMGMASK, BIMGMAX, BIMGBITS, BIMG2BITS
|
|
|
|
|
|
|
|
size_imageint = lmp_extract_setting(self, 'imageint')
|
|
|
|
IF (size_imageint == 8_c_int) THEN
|
|
|
|
IMGMASK = lmp_extract_setting(self, 'IMGMASK')
|
|
|
|
IMGMAX = lmp_extract_setting(self, 'IMGMAX')
|
|
|
|
IMGBITS = lmp_extract_setting(self, 'IMGBITS')
|
|
|
|
IMG2BITS = lmp_extract_setting(self, 'IMG2BITS')
|
|
|
|
BIMGMASK = IMGMASK
|
|
|
|
BIMGMAX = IMGMAX
|
|
|
|
BIMGBITS = IMGBITS
|
|
|
|
BIMG2BITS = IMG2BITS
|
2022-10-31 05:42:10 +08:00
|
|
|
flags(1) = INT(IAND(image, BIMGMASK) - BIMGMAX, KIND=c_int)
|
|
|
|
flags(2) = INT(IAND(ISHFT(image, -BIMGBITS), BIMGMASK) - BIMGMAX, &
|
|
|
|
KIND=c_int)
|
|
|
|
flags(3) = INT(ISHFT(image, -BIMG2BITS) - BIMGMAX, KIND=c_int)
|
2022-10-20 13:27:57 +08:00
|
|
|
ELSE
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, 'Incorrect&
|
|
|
|
& integer kind passed as "image" [Fortran/decode_image_flags]')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_decode_image_flags_bigbig
|
2022-10-20 07:40:12 +08:00
|
|
|
|
2022-12-23 12:09:20 +08:00
|
|
|
! constructor for fix_external_data that avoids a warning with Fortran 2003
|
|
|
|
! compilers
|
|
|
|
TYPE(fix_external_data) FUNCTION construct_fix_external_data()
|
2022-12-23 12:23:32 +08:00
|
|
|
construct_fix_external_data%id = ' '
|
2022-12-23 12:09:20 +08:00
|
|
|
END FUNCTION construct_fix_external_data
|
|
|
|
|
2022-11-22 12:38:10 +08:00
|
|
|
! equivalent function to lammps_set_fix_external_callback for -DSMALLSMALL
|
|
|
|
! note that "caller" is wrapped into a fix_external_data derived type along
|
|
|
|
! with the fix id and the Fortran calling function.
|
|
|
|
SUBROUTINE lmp_set_fix_external_callback(self, id, callback, caller)
|
|
|
|
CLASS(lammps), TARGET, INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
EXTERNAL :: callback
|
|
|
|
CLASS(*), INTENT(IN), TARGET, OPTIONAL :: caller
|
|
|
|
TYPE(c_ptr) :: c_id, c_caller
|
|
|
|
TYPE(c_funptr) :: c_callback
|
2022-11-25 11:07:46 +08:00
|
|
|
INTEGER :: i, this_fix
|
2022-11-22 12:38:10 +08:00
|
|
|
|
|
|
|
c_id = f2c_string(id)
|
|
|
|
IF (ALLOCATED(ext_data)) THEN
|
|
|
|
this_fix = SIZE(ext_data) + 1
|
|
|
|
DO i = 1, SIZE(ext_data)
|
|
|
|
IF (ext_data(i)%id == id) THEN
|
|
|
|
this_fix = i
|
|
|
|
EXIT
|
|
|
|
END IF
|
|
|
|
END DO
|
|
|
|
IF (this_fix > SIZE(ext_data)) THEN
|
2022-11-25 11:07:46 +08:00
|
|
|
! reallocates ext_data; this requires us to re-bind "caller" on the C
|
|
|
|
! side to the new data structure, which likely moved to a new address
|
2022-11-22 12:38:10 +08:00
|
|
|
ext_data = [ext_data, fix_external_data()] ! extends ext_data by 1
|
2022-11-25 11:07:46 +08:00
|
|
|
CALL rebind_external_callback_data()
|
2022-11-22 12:38:10 +08:00
|
|
|
END IF
|
|
|
|
ELSE
|
|
|
|
ALLOCATE(ext_data(1))
|
|
|
|
this_fix = 1
|
|
|
|
END IF
|
|
|
|
ext_data(this_fix)%id = id
|
|
|
|
ext_data(this_fix)%lammps_instance => self
|
|
|
|
|
2022-11-25 11:07:46 +08:00
|
|
|
IF (SIZE_TAGINT == 4_c_int .AND. SIZE_BIGINT == 4_c_int) THEN
|
2022-11-22 12:38:10 +08:00
|
|
|
! -DSMALLSMALL
|
|
|
|
c_callback = C_FUNLOC(callback_wrapper_smallsmall)
|
|
|
|
CALL set_fix_external_callback_smallsmall(this_fix, callback)
|
2022-11-25 11:07:46 +08:00
|
|
|
ELSE IF (SIZE_TAGINT == 8_c_int .AND. SIZE_BIGINT == 8_c_int) THEN
|
2022-11-22 12:38:10 +08:00
|
|
|
! -DBIGBIG
|
|
|
|
c_callback = C_FUNLOC(callback_wrapper_bigbig)
|
|
|
|
CALL set_fix_external_callback_bigbig(this_fix, callback)
|
|
|
|
ELSE
|
|
|
|
! -DSMALLBIG
|
|
|
|
c_callback = C_FUNLOC(callback_wrapper_smallbig)
|
|
|
|
CALL set_fix_external_callback_smallbig(this_fix, callback)
|
|
|
|
END IF
|
|
|
|
|
|
|
|
IF (PRESENT(caller)) THEN
|
|
|
|
ext_data(this_fix)%caller => caller
|
|
|
|
ELSE
|
|
|
|
NULLIFY(ext_data(this_fix)%caller)
|
|
|
|
END IF
|
|
|
|
c_caller = C_LOC(ext_data(this_fix))
|
|
|
|
CALL lammps_set_fix_external_callback(self%handle, c_id, c_callback, &
|
|
|
|
c_caller)
|
2022-11-25 11:07:46 +08:00
|
|
|
CALL lammps_free(c_id)
|
2022-11-22 12:38:10 +08:00
|
|
|
END SUBROUTINE lmp_set_fix_external_callback
|
|
|
|
|
|
|
|
! Wrappers to assign callback pointers with explicit interfaces
|
|
|
|
SUBROUTINE set_fix_external_callback_smallsmall(id, callback)
|
|
|
|
INTEGER, INTENT(IN) :: id
|
|
|
|
PROCEDURE(external_callback_smallsmall) :: callback
|
|
|
|
|
|
|
|
ext_data(id)%callback_smallsmall => callback
|
|
|
|
END SUBROUTINE set_fix_external_callback_smallsmall
|
|
|
|
|
|
|
|
SUBROUTINE set_fix_external_callback_smallbig(id, callback)
|
|
|
|
INTEGER, INTENT(IN) :: id
|
|
|
|
PROCEDURE(external_callback_smallbig) :: callback
|
|
|
|
|
|
|
|
ext_data(id)%callback_smallbig => callback
|
|
|
|
END SUBROUTINE set_fix_external_callback_smallbig
|
|
|
|
|
|
|
|
SUBROUTINE set_fix_external_callback_bigbig(id, callback)
|
|
|
|
INTEGER, INTENT(IN) :: id
|
|
|
|
PROCEDURE(external_callback_bigbig) :: callback
|
|
|
|
|
|
|
|
ext_data(id)%callback_bigbig => callback
|
|
|
|
END SUBROUTINE set_fix_external_callback_bigbig
|
|
|
|
|
2022-11-25 11:07:46 +08:00
|
|
|
! subroutine that re-binds all external callback data after a reallocation
|
|
|
|
SUBROUTINE rebind_external_callback_data()
|
|
|
|
INTEGER :: i
|
|
|
|
TYPE(c_ptr) :: c_id, c_caller
|
|
|
|
TYPE(c_funptr) :: c_callback
|
|
|
|
|
|
|
|
DO i = 1, SIZE(ext_data) - 1
|
|
|
|
c_id = f2c_string(ext_data(i)%id)
|
|
|
|
c_caller = C_LOC(ext_data(i))
|
|
|
|
IF (SIZE_TAGINT == 4_c_int .AND. SIZE_BIGINT == 4_c_int) THEN
|
|
|
|
c_callback = C_FUNLOC(callback_wrapper_smallsmall)
|
|
|
|
ELSE IF (SIZE_TAGINT == 8_c_int .AND. SIZE_BIGINT == 8_c_int) THEN
|
|
|
|
c_callback = C_FUNLOC(callback_wrapper_bigbig)
|
|
|
|
ELSE
|
|
|
|
c_callback = C_FUNLOC(callback_wrapper_smallbig)
|
|
|
|
END IF
|
|
|
|
CALL lammps_set_fix_external_callback( &
|
|
|
|
ext_data(i)%lammps_instance%handle, c_id, c_callback, c_caller)
|
|
|
|
CALL lammps_free(c_id)
|
|
|
|
END DO
|
|
|
|
END SUBROUTINE rebind_external_callback_data
|
|
|
|
|
2022-11-22 12:38:10 +08:00
|
|
|
! companions to lmp_set_fix_external_callback to change interface
|
|
|
|
SUBROUTINE callback_wrapper_smallsmall(caller, timestep, nlocal, ids, x, &
|
|
|
|
fexternal) BIND(C)
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: caller
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: timestep
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: nlocal
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: ids, x, fexternal
|
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: x0, f0
|
|
|
|
INTEGER(c_int), DIMENSION(:), POINTER :: f_ids => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:,:), POINTER :: f_x => NULL(), &
|
|
|
|
f_fexternal => NULL()
|
|
|
|
TYPE(fix_external_data), POINTER :: f_caller => NULL()
|
|
|
|
|
|
|
|
CALL C_F_POINTER(ids, f_ids, [nlocal])
|
|
|
|
CALL C_F_POINTER(x, x0, [nlocal])
|
|
|
|
CALL C_F_POINTER(x0(1), f_x, [3, nlocal])
|
|
|
|
CALL C_F_POINTER(fexternal, f0, [nlocal])
|
|
|
|
CALL C_F_POINTER(f0(1), f_fexternal, [3, nlocal])
|
|
|
|
IF (C_ASSOCIATED(caller)) THEN
|
|
|
|
CALL C_F_POINTER(caller, f_caller)
|
|
|
|
CALL f_caller%callback_smallsmall(f_caller%caller, timestep, f_ids, &
|
|
|
|
f_x, f_fexternal)
|
|
|
|
ELSE
|
|
|
|
CALL lmp_error(f_caller%lammps_instance, &
|
|
|
|
LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Got null pointer from "caller"; this should never happen;&
|
|
|
|
& please report a bug')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE callback_wrapper_smallsmall
|
|
|
|
|
|
|
|
SUBROUTINE callback_wrapper_smallbig(caller, timestep, nlocal, ids, x, &
|
|
|
|
fexternal) BIND(C)
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: caller
|
|
|
|
INTEGER(c_int64_t), INTENT(IN), VALUE :: timestep
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: nlocal
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: ids, x, fexternal
|
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: x0, f0
|
|
|
|
INTEGER(c_int), DIMENSION(:), POINTER :: f_ids => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:,:), POINTER :: f_x => NULL(), &
|
|
|
|
f_fexternal => NULL()
|
|
|
|
TYPE(fix_external_data), POINTER :: f_caller => NULL()
|
|
|
|
|
|
|
|
CALL C_F_POINTER(ids, f_ids, [nlocal])
|
|
|
|
CALL C_F_POINTER(x, x0, [nlocal])
|
|
|
|
CALL C_F_POINTER(x0(1), f_x, [3, nlocal])
|
|
|
|
CALL C_F_POINTER(fexternal, f0, [nlocal])
|
|
|
|
CALL C_F_POINTER(f0(1), f_fexternal, [3, nlocal])
|
|
|
|
IF (C_ASSOCIATED(caller)) THEN
|
|
|
|
CALL C_F_POINTER(caller, f_caller)
|
|
|
|
CALL f_caller%callback_smallbig(f_caller%caller, timestep, f_ids, f_x, &
|
|
|
|
f_fexternal)
|
|
|
|
ELSE
|
|
|
|
CALL lmp_error(f_caller%lammps_instance, &
|
|
|
|
LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Got null pointer from "caller"; this should never happen;&
|
|
|
|
& please report a bug')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE callback_wrapper_smallbig
|
|
|
|
|
|
|
|
SUBROUTINE callback_wrapper_bigbig(caller, timestep, nlocal, ids, x, &
|
|
|
|
fexternal) BIND(C)
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: caller
|
|
|
|
INTEGER(c_int64_t), INTENT(IN), VALUE :: timestep
|
|
|
|
INTEGER(c_int), INTENT(IN), VALUE :: nlocal
|
|
|
|
TYPE(c_ptr), INTENT(IN), VALUE :: ids, x, fexternal
|
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: x0, f0
|
|
|
|
INTEGER(c_int64_t), DIMENSION(:), POINTER :: f_ids => NULL()
|
|
|
|
REAL(c_double), DIMENSION(:,:), POINTER :: f_x => NULL(), &
|
|
|
|
f_fexternal => NULL()
|
|
|
|
TYPE(fix_external_data), POINTER :: f_caller => NULL()
|
|
|
|
|
|
|
|
CALL C_F_POINTER(ids, f_ids, [nlocal])
|
|
|
|
CALL C_F_POINTER(x, x0, [nlocal])
|
|
|
|
CALL C_F_POINTER(x0(1), f_x, [3, nlocal])
|
|
|
|
CALL C_F_POINTER(fexternal, f0, [nlocal])
|
|
|
|
CALL C_F_POINTER(f0(1), f_fexternal, [3, nlocal])
|
|
|
|
IF (C_ASSOCIATED(caller)) THEN
|
|
|
|
CALL C_F_POINTER(caller, f_caller)
|
|
|
|
CALL f_caller%callback_bigbig(f_caller%caller, timestep, f_ids, f_x, &
|
|
|
|
f_fexternal)
|
|
|
|
ELSE
|
|
|
|
CALL lmp_error(f_caller%lammps_instance, &
|
|
|
|
LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Got null pointer from "caller"; this should never happen;&
|
|
|
|
& please report a bug')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE callback_wrapper_bigbig
|
|
|
|
|
2022-11-30 08:28:52 +08:00
|
|
|
! equivalent function to lammps_fix_external_get_force
|
|
|
|
FUNCTION lmp_fix_external_get_force(self, id) RESULT(fexternal)
|
2022-12-01 12:48:29 +08:00
|
|
|
CLASS(lammps), TARGET, INTENT(IN) :: self
|
2022-11-30 08:28:52 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
2022-12-01 12:48:29 +08:00
|
|
|
TYPE(lammps_fix_data) :: fexternal
|
2022-11-30 08:28:52 +08:00
|
|
|
TYPE(c_ptr) :: ptr, Cid
|
|
|
|
TYPE(c_ptr), DIMENSION(:), POINTER :: f
|
|
|
|
INTEGER(c_int) :: nmax
|
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
ptr = lammps_fix_external_get_force(self%handle, Cid)
|
|
|
|
nmax = lmp_extract_setting(self, 'nmax')
|
|
|
|
CALL C_F_POINTER(ptr, f, [nmax])
|
2022-12-01 12:48:29 +08:00
|
|
|
fexternal%datatype = DATA_DOUBLE_2D
|
|
|
|
fexternal%lammps_instance => self
|
|
|
|
CALL C_F_POINTER(f(1), fexternal%r64_mat, [3, nmax])
|
2022-11-30 08:28:52 +08:00
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END FUNCTION lmp_fix_external_get_force
|
|
|
|
|
2022-11-30 13:23:14 +08:00
|
|
|
SUBROUTINE lmp_fix_external_set_energy_global(self, id, eng)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
2022-12-01 12:48:29 +08:00
|
|
|
REAL(c_double), INTENT(IN) :: eng
|
2022-11-30 13:23:14 +08:00
|
|
|
TYPE(c_ptr) :: Cid
|
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
CALL lammps_fix_external_set_energy_global(self%handle, Cid, eng)
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END SUBROUTINE lmp_fix_external_set_energy_global
|
|
|
|
|
|
|
|
SUBROUTINE lmp_fix_external_set_virial_global(self, id, virial)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
REAL(c_double), DIMENSION(6), TARGET, INTENT(IN) :: virial
|
|
|
|
TYPE(c_ptr) :: Cid, Cvirial
|
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
Cvirial = C_LOC(virial(1))
|
|
|
|
CALL lammps_fix_external_set_virial_global(self%handle, Cid, Cvirial)
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END SUBROUTINE lmp_fix_external_set_virial_global
|
|
|
|
|
|
|
|
SUBROUTINE lmp_fix_external_set_energy_peratom(self, id, eng)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
REAL(c_double), DIMENSION(:), TARGET, INTENT(IN) :: eng
|
|
|
|
TYPE(c_ptr) :: Cid, Ceng
|
|
|
|
INTEGER(c_int) :: nlocal
|
|
|
|
|
|
|
|
nlocal = lmp_extract_setting(self, 'nlocal')
|
|
|
|
IF (SIZE(eng) < nlocal) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Array "eng" should be length nlocal or greater &
|
|
|
|
&[Fortran/fix_external_set_energy_peratom]')
|
|
|
|
END IF
|
|
|
|
Cid = f2c_string(id)
|
2022-12-23 11:01:48 +08:00
|
|
|
Ceng = C_LOC(eng(1))
|
2022-11-30 13:23:14 +08:00
|
|
|
CALL lammps_fix_external_set_energy_peratom(self%handle, Cid, Ceng)
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END SUBROUTINE lmp_fix_external_set_energy_peratom
|
|
|
|
|
|
|
|
SUBROUTINE lmp_fix_external_set_virial_peratom(self, id, virial)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
REAL(c_double), DIMENSION(:,:), TARGET, INTENT(IN) :: virial
|
|
|
|
TYPE(c_ptr) :: Cid, Cvirial
|
2022-12-02 13:49:17 +08:00
|
|
|
TYPE(c_ptr), DIMENSION(:), ALLOCATABLE, TARGET :: Cptr
|
|
|
|
INTEGER(c_int) :: nlocal, i
|
2022-11-30 13:23:14 +08:00
|
|
|
|
|
|
|
nlocal = lmp_extract_setting(self, 'nlocal')
|
|
|
|
IF (SIZE(virial,2) < nlocal .OR. SIZE(virial,1) /= 6) THEN
|
|
|
|
CALL lmp_error(self, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'Array "virial" should be size 6 x nlocal or greater &
|
|
|
|
&[Fortran/fix_external_set_energy_peratom]')
|
|
|
|
END IF
|
|
|
|
Cid = f2c_string(id)
|
2022-12-02 13:49:17 +08:00
|
|
|
ALLOCATE(Cptr(nlocal))
|
|
|
|
DO i = 1, nlocal
|
|
|
|
Cptr(i) = C_LOC(virial(1,i))
|
|
|
|
END DO
|
|
|
|
Cvirial = C_LOC(Cptr(1))
|
2022-11-30 13:23:14 +08:00
|
|
|
CALL lammps_fix_external_set_virial_peratom(self%handle, Cid, Cvirial)
|
|
|
|
CALL lammps_free(Cid)
|
2022-12-02 13:49:17 +08:00
|
|
|
DEALLOCATE(Cptr)
|
2022-11-30 13:23:14 +08:00
|
|
|
END SUBROUTINE lmp_fix_external_set_virial_peratom
|
|
|
|
|
|
|
|
SUBROUTINE lmp_fix_external_set_vector_length(self, id, length)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
INTEGER(c_int), INTENT(IN) :: length
|
|
|
|
TYPE(c_ptr) :: Cid
|
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
CALL lammps_fix_external_set_vector_length(self%handle, Cid, length)
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END SUBROUTINE lmp_fix_external_set_vector_length
|
|
|
|
|
|
|
|
SUBROUTINE lmp_fix_external_set_vector(self, id, idx, val)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: id
|
|
|
|
INTEGER(c_int), INTENT(IN) :: idx
|
|
|
|
REAL(c_double), INTENT(IN) :: val
|
|
|
|
TYPE(c_ptr) :: Cid
|
|
|
|
|
|
|
|
Cid = f2c_string(id)
|
|
|
|
CALL lammps_fix_external_set_vector(self%handle, Cid, idx, val)
|
|
|
|
CALL lammps_free(Cid)
|
|
|
|
END SUBROUTINE lmp_fix_external_set_vector
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
! equivalent function to lammps_flush_buffers
|
|
|
|
SUBROUTINE lmp_flush_buffers(self)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
|
|
|
|
CALL lammps_flush_buffers(self%handle)
|
|
|
|
END SUBROUTINE lmp_flush_buffers
|
|
|
|
|
2022-09-28 20:40:06 +08:00
|
|
|
! equivalent function to lammps_is_running
|
|
|
|
LOGICAL FUNCTION lmp_is_running(self)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
lmp_is_running = (lammps_is_running(self%handle) /= 0_c_int)
|
2022-09-28 20:40:06 +08:00
|
|
|
END FUNCTION lmp_is_running
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
! equivalent function to lammps_force_timeout
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE lmp_force_timeout(self)
|
2022-09-28 20:40:06 +08:00
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
|
2022-09-30 08:28:15 +08:00
|
|
|
CALL lammps_force_timeout(self%handle)
|
|
|
|
END SUBROUTINE
|
2022-09-28 20:40:06 +08:00
|
|
|
|
2022-09-26 21:57:03 +08:00
|
|
|
! equivalent function to lammps_has_error
|
|
|
|
LOGICAL FUNCTION lmp_has_error(self)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
2022-09-27 10:56:03 +08:00
|
|
|
INTEGER(c_int) :: has_error
|
2022-09-26 21:57:03 +08:00
|
|
|
|
|
|
|
has_error = lammps_has_error(self%handle)
|
2022-09-27 10:56:03 +08:00
|
|
|
lmp_has_error = (has_error /= 0_c_int)
|
2022-09-26 21:57:03 +08:00
|
|
|
END FUNCTION lmp_has_error
|
|
|
|
|
2022-09-27 10:56:03 +08:00
|
|
|
! equivalent function to lammps_get_last_error_message
|
|
|
|
SUBROUTINE lmp_get_last_error_message(self, buffer, status)
|
|
|
|
CLASS(lammps), INTENT(IN) :: self
|
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: buffer
|
|
|
|
INTEGER, INTENT(OUT), OPTIONAL :: status
|
2022-10-05 22:52:17 +08:00
|
|
|
INTEGER(c_int) :: buflen, Cstatus
|
2022-10-28 03:12:26 +08:00
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(LEN(buffer)+1), TARGET :: Cbuffer
|
2022-09-29 11:09:26 +08:00
|
|
|
TYPE(c_ptr) :: Cptr
|
2022-09-27 10:56:03 +08:00
|
|
|
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (lmp_has_error(self)) THEN
|
2022-10-28 03:12:26 +08:00
|
|
|
buflen = LEN(buffer, KIND=c_int) + 1_c_int
|
|
|
|
Cptr = C_LOC(Cbuffer(1))
|
2022-10-04 11:18:44 +08:00
|
|
|
Cstatus = lammps_get_last_error_message(self%handle, Cptr, buflen)
|
2022-10-28 03:12:26 +08:00
|
|
|
buffer = array2string(Cbuffer)
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(status)) THEN
|
2022-09-27 10:56:03 +08:00
|
|
|
status = Cstatus
|
|
|
|
END IF
|
|
|
|
ELSE
|
2022-12-23 12:23:32 +08:00
|
|
|
buffer = ' '
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (PRESENT(status)) THEN
|
2022-09-27 10:56:03 +08:00
|
|
|
status = 0
|
|
|
|
END IF
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE lmp_get_last_error_message
|
|
|
|
|
2022-08-12 13:04:25 +08:00
|
|
|
! ----------------------------------------------------------------------
|
|
|
|
! functions to assign user-space pointers to LAMMPS data
|
|
|
|
! ----------------------------------------------------------------------
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_int_to_lammps_data(lhs, rhs)
|
2022-08-12 13:04:25 +08:00
|
|
|
INTEGER(c_int), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_INT) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
lhs => rhs%i32
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'scalar int')
|
2022-08-12 13:04:25 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_int_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_int64_to_lammps_data(lhs, rhs)
|
2022-08-12 13:04:25 +08:00
|
|
|
INTEGER(c_int64_t), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_INT64) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
lhs => rhs%i64
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'scalar long int')
|
2022-08-12 13:04:25 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_int64_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_intvec_to_lammps_data(lhs, rhs)
|
2022-09-15 10:07:32 +08:00
|
|
|
INTEGER(c_int), DIMENSION(:), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_INT_1D) THEN
|
2022-09-15 10:07:32 +08:00
|
|
|
lhs => rhs%i32_vec
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'vector of ints')
|
2022-09-15 10:07:32 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_intvec_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_int64vec_to_lammps_data(lhs, rhs)
|
2022-09-23 08:16:15 +08:00
|
|
|
INTEGER(c_int64_t), DIMENSION(:), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_INT64_1D) THEN
|
2022-09-23 08:16:15 +08:00
|
|
|
lhs => rhs%i64_vec
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'vector of long ints')
|
2022-09-23 08:16:15 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_int64vec_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_double_to_lammps_data(lhs, rhs)
|
2022-08-12 13:04:25 +08:00
|
|
|
REAL(c_double), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
lhs => rhs%r64
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'scalar double')
|
2022-08-12 13:04:25 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_double_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_doublevec_to_lammps_data(lhs, rhs)
|
2022-08-12 13:04:25 +08:00
|
|
|
REAL(c_double), DIMENSION(:), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE_1D) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
lhs => rhs%r64_vec
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'vector of doubles')
|
2022-08-12 13:04:25 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_doublevec_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_doublemat_to_lammps_data(lhs, rhs)
|
2022-09-23 08:16:15 +08:00
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE_2D) THEN
|
2022-09-23 08:16:15 +08:00
|
|
|
lhs => rhs%r64_mat
|
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'matrix of doubles')
|
2022-09-23 08:16:15 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_doublemat_to_lammps_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_string_to_lammps_data(lhs, rhs)
|
2022-08-12 13:04:25 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: lhs
|
|
|
|
CLASS(lammps_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_STRING) THEN
|
2022-08-12 13:04:25 +08:00
|
|
|
lhs = rhs%str
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (LEN_TRIM(rhs%str) > LEN(lhs)) THEN
|
2022-10-04 06:44:09 +08:00
|
|
|
CALL lmp_error(rhs%lammps_instance, LMP_ERROR_WARNING, &
|
|
|
|
'String provided by user required truncation [Fortran API]')
|
|
|
|
END IF
|
2022-08-12 13:04:25 +08:00
|
|
|
ELSE
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL assignment_error(rhs, 'string')
|
2022-08-12 13:04:25 +08:00
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_string_to_lammps_data
|
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
! ----------------------------------------------------------------------
|
|
|
|
! functions to assign user-space pointers to LAMMPS *fix* data
|
|
|
|
! ----------------------------------------------------------------------
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_double_to_lammps_fix_data(lhs, rhs)
|
2022-09-29 11:09:26 +08:00
|
|
|
REAL(c_double), INTENT(OUT) :: lhs
|
|
|
|
CLASS(lammps_fix_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
lhs = rhs%r64
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'scalar double')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_double_to_lammps_fix_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_doublevec_to_lammps_fix_data(lhs, rhs)
|
2022-09-29 11:09:26 +08:00
|
|
|
REAL(c_double), DIMENSION(:), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_fix_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE_1D) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
lhs => rhs%r64_vec
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'vector of doubles')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_doublevec_to_lammps_fix_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_doublemat_to_lammps_fix_data(lhs, rhs)
|
2022-09-29 11:09:26 +08:00
|
|
|
REAL(c_double), DIMENSION(:,:), INTENT(OUT), POINTER :: lhs
|
|
|
|
CLASS(lammps_fix_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE_2D) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
lhs => rhs%r64_mat
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'matrix of doubles')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_doublemat_to_lammps_fix_data
|
|
|
|
|
|
|
|
! ----------------------------------------------------------------------
|
|
|
|
! functions to assign user-space pointers to LAMMPS *variable* data
|
|
|
|
! ----------------------------------------------------------------------
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_double_to_lammps_variable_data(lhs, rhs)
|
2022-09-29 11:09:26 +08:00
|
|
|
REAL(c_double), INTENT(OUT) :: lhs
|
|
|
|
CLASS(lammps_variable_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE) THEN
|
2022-09-29 11:09:26 +08:00
|
|
|
lhs = rhs%r64
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'scalar double')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_double_to_lammps_variable_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_doublevec_to_lammps_variable_data(lhs, rhs)
|
2022-10-01 05:02:26 +08:00
|
|
|
REAL(c_double), DIMENSION(:), ALLOCATABLE, INTENT(OUT) :: lhs
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps_variable_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_DOUBLE_1D) THEN
|
|
|
|
IF (ALLOCATED(lhs)) DEALLOCATE(lhs)
|
|
|
|
ALLOCATE(lhs(SIZE(rhs%r64_vec)))
|
2022-09-29 11:09:26 +08:00
|
|
|
lhs = rhs%r64_vec
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'vector of doubles')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_doublevec_to_lammps_variable_data
|
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assign_string_to_lammps_variable_data(lhs, rhs)
|
2022-09-30 08:28:15 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(OUT) :: lhs
|
|
|
|
CLASS(lammps_variable_data), INTENT(IN) :: rhs
|
|
|
|
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (rhs%datatype == DATA_STRING) THEN
|
2022-09-30 08:28:15 +08:00
|
|
|
lhs = rhs%str
|
2022-10-04 11:49:17 +08:00
|
|
|
IF (LEN_TRIM(rhs%str) > LEN(lhs)) THEN
|
2022-10-04 06:44:09 +08:00
|
|
|
CALL lmp_error(rhs%lammps_instance, LMP_ERROR_WARNING, &
|
|
|
|
'String provided by user required truncation [Fortran API]')
|
|
|
|
END IF
|
2022-09-30 08:28:15 +08:00
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'string')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_string_to_lammps_variable_data
|
|
|
|
|
2022-10-20 07:40:12 +08:00
|
|
|
SUBROUTINE assign_int_to_lammps_image_data(lhs, rhs)
|
|
|
|
INTEGER(c_int), INTENT(OUT) :: lhs
|
|
|
|
CLASS(lammps_image_data), INTENT(IN) :: rhs
|
|
|
|
|
|
|
|
IF (rhs%datatype == DATA_INT) THEN
|
|
|
|
lhs = rhs%i32
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'scalar int')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_int_to_lammps_image_data
|
|
|
|
|
|
|
|
SUBROUTINE assign_int64_to_lammps_image_data(lhs, rhs)
|
|
|
|
INTEGER(c_int64_t), INTENT(OUT) :: lhs
|
|
|
|
CLASS(lammps_image_data), INTENT(IN) :: rhs
|
|
|
|
|
|
|
|
IF (rhs%datatype == DATA_INT64) THEN
|
|
|
|
lhs = rhs%i64
|
|
|
|
ELSE
|
|
|
|
CALL assignment_error(rhs, 'scalar long int')
|
|
|
|
END IF
|
|
|
|
END SUBROUTINE assign_int64_to_lammps_image_data
|
|
|
|
|
2022-09-29 11:09:26 +08:00
|
|
|
! ----------------------------------------------------------------------
|
|
|
|
! Generic function to catch all errors in assignments of LAMMPS data to
|
|
|
|
! user-space variables/pointers
|
|
|
|
! ----------------------------------------------------------------------
|
2022-10-01 07:50:16 +08:00
|
|
|
SUBROUTINE assignment_error(type1, str2)
|
2022-09-29 11:09:26 +08:00
|
|
|
CLASS(lammps_data_baseclass), INTENT(IN) :: type1
|
2022-10-01 07:50:16 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: str2
|
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: str1
|
2022-09-15 10:07:32 +08:00
|
|
|
|
2022-10-01 07:50:16 +08:00
|
|
|
SELECT CASE(type1%datatype)
|
|
|
|
CASE(DATA_INT)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'scalar int'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_INT_1D)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'vector of ints'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_INT_2D)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'matrix of ints'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_INT64)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'scalar long int'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_INT64_1D)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'vector of long ints'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_INT64_2D)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'matrix of long ints'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_DOUBLE)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'scalar double'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_DOUBLE_1D)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'vector of doubles'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_DOUBLE_2D)
|
2022-09-15 10:07:32 +08:00
|
|
|
str1 = 'matrix of doubles'
|
2022-10-01 07:50:16 +08:00
|
|
|
CASE(DATA_STRING)
|
2022-09-30 08:28:15 +08:00
|
|
|
str1 = 'string'
|
2022-09-15 10:07:32 +08:00
|
|
|
CASE DEFAULT
|
|
|
|
str1 = 'that type'
|
|
|
|
END SELECT
|
2022-09-29 11:09:26 +08:00
|
|
|
CALL lmp_error(type1%lammps_instance, LMP_ERROR_ALL + LMP_ERROR_WORLD, &
|
|
|
|
'cannot associate ' // str1 // ' with ' // str2 // ' [Fortran API]')
|
2022-09-15 10:07:32 +08:00
|
|
|
END SUBROUTINE assignment_error
|
2022-08-12 13:04:25 +08:00
|
|
|
|
2020-08-29 08:56:15 +08:00
|
|
|
! ----------------------------------------------------------------------
|
|
|
|
! local helper functions
|
|
|
|
! copy fortran string to zero terminated c string
|
2022-08-07 05:43:40 +08:00
|
|
|
! ----------------------------------------------------------------------
|
2020-08-29 08:56:15 +08:00
|
|
|
FUNCTION f2c_string(f_string) RESULT(ptr)
|
2022-08-12 13:04:25 +08:00
|
|
|
CHARACTER(LEN=*), INTENT(IN) :: f_string
|
|
|
|
CHARACTER(LEN=1, KIND=c_char), POINTER :: c_string(:)
|
2020-08-29 08:56:15 +08:00
|
|
|
TYPE(c_ptr) :: ptr
|
2020-10-02 22:36:17 +08:00
|
|
|
INTEGER(c_size_t) :: i, n
|
2020-08-29 08:56:15 +08:00
|
|
|
|
|
|
|
n = LEN_TRIM(f_string)
|
2020-10-02 22:36:17 +08:00
|
|
|
ptr = lammps_malloc(n+1)
|
2024-03-18 22:27:38 +08:00
|
|
|
CALL C_F_POINTER(ptr, c_string, [n+1])
|
2022-08-07 05:43:40 +08:00
|
|
|
DO i=1, n
|
2020-08-29 08:56:15 +08:00
|
|
|
c_string(i) = f_string(i:i)
|
|
|
|
END DO
|
|
|
|
c_string(n+1) = c_null_char
|
|
|
|
END FUNCTION f2c_string
|
2022-10-21 09:24:59 +08:00
|
|
|
|
2022-10-26 08:51:08 +08:00
|
|
|
! copy null-terminated C string to fortran string
|
|
|
|
FUNCTION c2f_string(ptr) RESULT(f_string)
|
|
|
|
TYPE(c_ptr), INTENT(IN) :: ptr
|
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: f_string
|
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(:), POINTER :: c_string
|
2022-10-28 03:12:26 +08:00
|
|
|
INTEGER :: n
|
|
|
|
|
2022-10-26 08:51:08 +08:00
|
|
|
IF (.NOT. C_ASSOCIATED(ptr)) THEN
|
2022-12-23 12:23:32 +08:00
|
|
|
f_string = ' '
|
2022-11-01 01:05:53 +08:00
|
|
|
ELSE
|
|
|
|
n = INT(c_strlen(ptr), KIND=KIND(n))
|
|
|
|
CALL C_F_POINTER(ptr, c_string, [n+1])
|
|
|
|
f_string = array2string(c_string, n)
|
2022-10-26 08:51:08 +08:00
|
|
|
END IF
|
2022-10-28 03:12:26 +08:00
|
|
|
END FUNCTION c2f_string
|
|
|
|
|
|
|
|
! Copy a known-length or null-terminated array of C characters into a string
|
|
|
|
FUNCTION array2string(array, length) RESULT(string)
|
|
|
|
CHARACTER(LEN=1, KIND=c_char), DIMENSION(:) :: array
|
|
|
|
! NOTE: giving "length" the VALUE attribute reveals a bug in gfortran 12.2.1
|
|
|
|
! https://gcc.gnu.org/bugzilla/show_bug.cgi?id=107441
|
|
|
|
INTEGER, INTENT(IN), OPTIONAL :: length
|
|
|
|
CHARACTER(LEN=:), ALLOCATABLE :: string
|
|
|
|
INTEGER :: n, i
|
|
|
|
|
|
|
|
IF (PRESENT(length)) THEN
|
|
|
|
n = length
|
|
|
|
ELSE
|
|
|
|
n = 1
|
|
|
|
DO WHILE (n < SIZE(array) .AND. array(n+1) /= c_null_char)
|
|
|
|
n = n + 1
|
|
|
|
END DO
|
|
|
|
END IF
|
|
|
|
ALLOCATE(CHARACTER(LEN=n) :: string)
|
2022-10-26 08:51:08 +08:00
|
|
|
DO i = 1, n
|
2022-10-28 03:12:26 +08:00
|
|
|
string(i:i) = array(i)
|
2022-10-26 08:51:08 +08:00
|
|
|
END DO
|
2022-10-28 03:12:26 +08:00
|
|
|
END FUNCTION array2string
|
2022-10-26 08:51:08 +08:00
|
|
|
|
2020-08-29 08:56:15 +08:00
|
|
|
END MODULE LIBLAMMPS
|
2022-08-12 06:46:21 +08:00
|
|
|
|
|
|
|
! vim: ts=2 sts=2 sw=2 et
|